Member function pointer and high performance C ++ entrusted (middle)

xiaoxiao2021-03-05  20

Member Function Pointers and the Fastest Possible C Delegates Writes: DON Clugston Don Clugston Translation: Zhou Xiang (connected) Member function pointer - Why is so complicated? The member functions of the class and the standard C function are different. Similar to the parameters explicitly declared, the member function of the class has a hidden parameter this, which points to an instance of a class. Depending on the compiler, this or a normal parameter is considered inside, or will be specially treated (for example, in VC , this is generally passed through the ECX register, and the parameters of the ordinary member function are directly pressed in the stack in). This is the essence of the parameters and other ordinary parameters. Even if a member function is dominated by a normal function, there is no reason to make this member function and other ordinary functions in standard C , because no ThisCall keyword guarantees that it uses the normal parameters as normal call rules. The member function is a matter, the normal function is another thing (MEMBER FUNCTIONS ARE from Venus). You may guess, like a member function pointer and a normal function pointer, just a code pointer. However, this guess may be wrong. In most compilers, a member function pointer is much larger than a normal function pointer. Surprisingly, in Visual C , a member function pointer can be 4, 8, 12 or even 16 bytes long, depending on the nature of the class it related, but also depends on what kind of compilation used by the compiler. Set! Member function pointer is much more complicated than you imagined, but it is not always like this. Let us return to the early 1980s, then, the oldest C compiler cfront has just been developed, then the C language can only achieve single inheritance, and the member function pointer is just introduced, and they are very simple: they are like Ordinary function pointers, only additional this as their first parameters, you can convert a member function pointer into a normal function pointer and enable you to pay sufficient attention to this additional parameter. This idyllic world has been smashed with CFRONT 2.0. It introduces template and multiple inheritance, and the damage caused by multiple inherits causes a change in member function pointer. The problem is that with multiple inherits, you don't know which pointer to use which parent class is used, for example, you have 4 classes to be as follows: C functions have some differences. Similar to the parameters explicitly declared, the member function of the class has a hidden parameter this, which points to an instance of a class. Depending on the compiler, this or a normal parameter is considered inside, or will be specially treated (for example, in VC , this is generally passed through the ECX register, and the parameters of the ordinary member function are directly pressed in the stack in). This is the essence of the parameters and other ordinary parameters. Even if a member function is dominated by a normal function, there is no reason to make this member function and other ordinary functions in standard C , because no ThisCall keyword guarantees that it uses the normal parameters as normal call rules.

The member function is a matter, the normal function is another thing (MEMBER FUNCTIONS ARE from Venus). You may guess, like a member function pointer and a normal function pointer, just a code pointer. However, this guess may be wrong. In most compilers, a member function pointer is much larger than a normal function pointer. Surprisingly, in Visual C , a member function pointer can be 4, 8, 12 or even 16 bytes long, depending on the nature of the class it related, but also depends on what kind of compilation used by the compiler. Set! Member function pointer is much more complicated than you imagined, but it is not always like this. Let us return to the early 1980s, then, the oldest C compiler cfront has just been developed, then the C language can only achieve single inheritance, and the member function pointer is just introduced, and they are very simple: they are like Ordinary function pointers, only additional this as their first parameters, you can convert a member function pointer into a normal function pointer and enable you to pay sufficient attention to this additional parameter. This idyllic world has been smashed with CFRONT 2.0. It introduces template and multiple inheritance, and the damage caused by multiple inherits causes a change in member function pointer. The problem is that with multiple inherits, you don't know which pointer to use which parent class is used, for example, you have 4 classes as follows: Visual C , a member function pointer can be 4, 8, 12 or even 16 The byte is long, depending on the nature of the class it related, but also depends on what kind of compilation settings used in the compiler! Member function pointer is much more complicated than you imagined, but it is not always like this. Let us return to the early 1980s, then, the oldest C compiler cfront has just been developed, then the C language can only achieve single inheritance, and the member function pointer is just introduced, and they are very simple: they are like Ordinary function pointers, only additional this as their first parameters, you can convert a member function pointer into a normal function pointer and enable you to pay sufficient attention to this additional parameter. This idyllic world has been smashed with CFRONT 2.0. It introduces template and multiple inheritance, and the damage caused by multiple inherits causes a change in member function pointer. The problem is that with multiple inheritance, before calling, you don't know which parent class of the THIS pointer, for example, you have 4 categories as follows: At the beginning of the 1980s, the oldest C compiler cfront has just been developed, At that time, the C language only implemented a single inherit, and the member function pointer was introduced, and they were simple: they were like a normal function pointer, just adding additional this as their first parameters, you can file a member function The pointer is transformed into a normal function pointer and enables you to pay sufficient attention to this additional parameter. This idyllic world has been smashed with CFRONT 2.0. It introduces template and multiple inheritance, and the damage caused by multiple inherits causes a change in member function pointer. The problem is that with multiple inherits, you don't know which pointer to use which parent class is used, for example, you have 4 classes as follows: CFRONT 2.0 has been smashed. It introduces template and multiple inheritance, and the damage caused by multiple inherits causes a change in member function pointer.

The problem is that with multiple inheritance, before calling, you don't know which parent class of this pointer, for example, you have 4 classes defined as follows: Class A {public: Virtual int Afunc () {return 2;}; Class B {public: int bfunc () {return 3;};}; // c is a single inheritance class, which is only inherited in ACLASS C: public a {public: int cFunc () {Return 4;};}; / The / D class uses multiple inheritance Class D: public a, public b {public: int DFUNC () {Return 5;};}; if we have a member function pointer for Class C. In this example, both AFUNC and CFUNC are members of C, so our member function pointer can point to AFUNC or CFUNC. But AFUNC requires a THIS pointer to C :: A (later I call it athis), and cfunc requires a THIS pointer to c (later I call it Cthis). The compiler designer uses a trick (Trick) to handle this situation: they guarantee that the Class A is physically stored in the head of Class C (that is, the Class C start address is an example of a class A). The start address), which means ATHIS == CTHIS. We only need to worry about a THIS pointer is enough, and for the current situation, all issues can be handled. Now, if we create a member function pointer of a class D class. In this case, our member function pointer can point to AFUNC, BFUNC or DFUNC. But AFUNC requires a THIS pointer to D :: A, while BFUNC requires a THIS pointer points to D :: B. At this time, this trick doesn't care, we can't put the A and C class in the head of the D class. Therefore, one member function pointer of class D is not only to indicate which function is to specify the call, but also specify which THIS pointer to use. The compiler knows how big is the space occupied by Class A, so it can add an ATHIS pointer to the BTHIS pointer to ATHIS. If you use virtual inheritance, such as virtual base classes, the situation will become worse, you don't have to understand why this is too hurt. Just give an example, the compiler uses the virtual function table (Virtual Function Table - "VTable") to save each virtual function, the address of the function, the address and Virtual_Delta: Convert the current THIS pointer to the actual function. The displacement amount to be added when the THIS pointer is added. In summary, in order to support the general form of member function pointer, you need to at least three information: the address of the function, you need to increase the Delta bit shift on the THIS pointer, and the index in a virtual function table. For MSVC, you need a fourth message: The address of the virtual function table (VTABLE). A member function pointer for Class C. In this example, both AFUNC and CFUNC are members of C, so our member function pointer can point to AFUNC or CFUNC. But AFUNC requires a THIS pointer to C :: A (later I call it athis), and cfunc requires a THIS pointer to c (later I call it Cthis).

The compiler designer uses a trick (Trick) to handle this situation: they guarantee that the Class A is physically stored in the head of Class C (that is, the Class C start address is an example of a class A). The start address), which means ATHIS == CTHIS. We only need to worry about a THIS pointer is enough, and for the current situation, all issues can be handled. Now, if we create a member function pointer of a class D class. In this case, our member function pointer can point to AFUNC, BFUNC or DFUNC. But AFUNC requires a THIS pointer to D :: A, while BFUNC requires a THIS pointer points to D :: B. At this time, this trick doesn't care, we can't put the A and C class in the head of the D class. Therefore, one member function pointer of class D is not only to indicate which function is to specify the call, but also specify which THIS pointer to use. The compiler knows how big is the space occupied by Class A, so it can add an ATHIS pointer to the BTHIS pointer to ATHIS. If you use virtual inheritance, such as virtual base classes, the situation will become worse, you don't have to understand why this is too hurt. Just give an example, the compiler uses the virtual function table (Virtual Function Table - "VTable") to save each virtual function, the address of the function, the address and Virtual_Delta: Convert the current THIS pointer to the actual function. The displacement amount to be added when the THIS pointer is added. In summary, in order to support the general form of member function pointer, you need to at least three information: the address of the function, you need to increase the Delta bit shift on the THIS pointer, and the index in a virtual function table. For MSVC, you need a fourth message: The address of the virtual function table (VTABLE). Class D member function pointer. In this case, our member function pointer can point to AFUNC, BFUNC or DFUNC. But AFUNC requires a THIS pointer to D :: A, while BFUNC requires a THIS pointer points to D :: B. At this time, this trick doesn't care, we can't put the A and C class in the head of the D class. Therefore, one member function pointer of class D is not only to indicate which function is to specify the call, but also specify which THIS pointer to use. The compiler knows how big is the space occupied by Class A, so it can add an ATHIS pointer to the BTHIS pointer to ATHIS. If you use virtual inheritance, such as virtual base classes, the situation will become worse, you don't have to understand why this is too hurt. Just give an example, the compiler uses the virtual function table (Virtual Function Table - "VTable") to save each virtual function, the address of the function, the address and Virtual_Delta: Convert the current THIS pointer to the actual function. The displacement amount to be added when the THIS pointer is added. In summary, in order to support the general form of member function pointer, you need to at least three information: the address of the function, you need to increase the Delta bit shift on the THIS pointer, and the index in a virtual function table. For MSVC, you need a fourth message: The address of the virtual function table (VTABLE). Virtual inheritance, such as vain, the situation will become worse, you don't have to understand why it is too hurtful.

Just give an example, the compiler uses the virtual function table (Virtual Function Table - "VTable") to save each virtual function, the address of the function, the address and Virtual_Delta: Convert the current THIS pointer to the actual function. The displacement amount to be added when the THIS pointer is added. In summary, in order to support the general form of member function pointer, you need to at least three information: the address of the function, you need to increase the Delta bit shift on the THIS pointer, and the index in a virtual function table. For MSVC, you need a fourth message: The address of the virtual function table (VTABLE). The Delta displacement on the this pointer, and the index in a virtual function table. For MSVC, you need a fourth message: The address of the virtual function table (VTABLE). What is the implementation of a member function pointer? How does the compiler implement the member function pointer? Here is a compiler for different 32, 64 and 16 bits, for a variety of different data types (with int, void * data pointers, code pointers (such as pointers pointing to the static function), inherited in a single (SINGLE-), Multiple-) Inheritance, Virtual- Various data types (with int, void * data pointers, code pointers (such as pointers pointing to static functions), inheritance, multiple (multiple-) inheritance, virtual-) inheritance and unknown type The member function pointer of the class under the inheritance of unknown) Use the SizeOF operator to calculate the data obtained:

Compiler option intDataPtrCodePtrSingleMultiVirtualUnknownMSVC444481216MSVC / vmg44416 # 16 # 16 # 16MSVC / vmg / vmm4448 # 8 # - 8 # Intel_IA32444481212Intel_IA32 / vmg / vmm44448--8Intel_Itanium4888122020G 4448888Comeau4448888DMC4444444BCC3244412121212BCC32 / Vmd444481212WCL38644412121212CodeWarrior44412121212XLC48820202020DMCsmall2222222DMCmedium2244444WCLsmall2226666WCLcompact2426666WCLmedium2248888WCLlarge2448888 Note: # indicates use __single / __ multi / __ virtual_inheritance keywords Time represents 4, 8 or 12. These compilers are Microsoft Visual C 4.0 to 7.1 (.NET 2003), GNU G 3.2 (Mingw Binaries, http://www.mingw.org/), Borland BCB 5.1 (http://www.borland.com/) Open Watcom (WCL) 1.2 (http://www.openwatcom.org/), DIGITAL MARS (DMC) 8.38N (http://www.digitalmars.com/), Intel C 8.0 for Windows IA-32, Intel C 8.0 for itanium, (http://www.intel.com/), IBM XLC for AIX (Power, PowerPC), Metrowerks Code Warrior 9.1 for Windows (http://www.metrowerks.com/), and Comeau C 4.3 (http://www.comeaucomputing.com/). Comeau's data is obtained on its supported 32-bit platform (X86, Alpha, SPARC, etc.). The 16-bit compiler is tested under four DOS configurations (Tiny, Compact, MEDIUM, and LARGE), it is used to display a variety of different code and data pointers. The MSVC is tested under / vmg, used to display all the features of the member pointer. (If you have a compiler in the list, please let me know. The compiler test results under the non-X86 processor have a unique value.) Indicates that when using the __single / __ multi / __ virtual_inheritance keyword, represents 4, 8 or 12.

These compilers are Microsoft Visual C 4.0 to 7.1 (.NET 2003), GNU G 3.2 (Mingw Binaries, http://www.mingw.org/), Borland BCB 5.1 (http://www.borland.com/) Open Watcom (WCL) 1.2 (http://www.openwatcom.org/), DIGITAL MARS (DMC) 8.38N (http://www.digitalmars.com/), Intel C 8.0 for Windows IA-32, Intel C 8.0 for itanium, (http://www.intel.com/), IBM XLC for AIX (Power, PowerPC), Metrowerks Code Warrior 9.1 for Windows (http://www.metrowerks.com/), and Comeau C 4.3 (http://www.comeaucomputing.com/). Comeau's data is obtained on its supported 32-bit platform (X86, Alpha, SPARC, etc.). The 16-bit compiler is tested under four DOS configurations (Tiny, Compact, MEDIUM, and LARGE), it is used to display a variety of different code and data pointers. The MSVC is tested under / vmg, used to display all the features of the member pointer. (If you have a compiler in the list, please let me know. The compiler test results under the non-x86 processor have a unique value.) Microsoft Visual C 4.0 to 7.1 (.NET 2003), GNU G 3.2 (MINGW Binaries, http://www.mingw.org/), Borland BCB 5.1 (http://www.borland.com/), Open Watcom (WCL) 1.2 (http://www.openwatcom.org/), DIGITAL MARS (DMC) 8.38N (http://www.digitalmars.com/), Intel C 8.0 for Windows IA-32, Intel C 8.0 for itanium, (http://www.intel.com/), IBM XLC for AIX (Power, PowerPC), Metrowerks Code Warrior 9.1 for Windows (http://www.metrowerks.com/), and Comeau C 4.3 (http://www.comeaucomputing.com/). Comeau's data is in it support 32-bit platforms (X86, Alpha, SPARC, etc.) are obtained. The 16-bit compiler is tested under four DOS configurations (Tiny, Compact, MEDIUM, and LARGE), it is used to display a variety of different code and data pointers. The MSVC is tested under / vmg, used to display all the features of the member pointer. (If you have a compiler in the list, please let me know. The compiler test results under the non-x86 processor have a unique value.) Look at the data in the table, do you think is very surprised? You can clearly see the code written in some environments and cannot run in other compilers.

Between different compilers, their internal implementation is obviously very different; in fact, I think the compiler does not have a significant difference in other characteristics of the language. Studying the details of the implementation You will find some strange problems. Generally, the compiler takes the worst, and has always used the most common form. For example, for this structure: // borland (default setting) and Watcom C . Struct {functionpointer m_func_address; int m_delta; int m_vtable_index; // If it is not a virtual inheritance, this value is 0. };}; // Metrowerks CodeWarrior uses a slightly different way. // Even if you do not allow multiple inherited Embedded C mode, it also uses this structure! Struct {int m_delta; int m_vtable_index; // If it is not a virtual inheritance, this value is -1. FunctionPointer m_func_address;}; // An early Suncc version obviously uses another rule: struct {int m_vtable_index; // If it is a non-virtual function, this value is 0. FunctionPointer m_func_address; // If it is a virtual function (Virtual function), this value is 0. INT M_DELTA;}; // The following is the method of Microsoft's compiler in the case of unknown inheritance or using the / VMG option: struct {functionpointer m_func_address; int m_delta; int m_vtordisp; int m_vtable_index; // If not virtual inheritance This value is 0}; // AIX (PowerPC) on IBM XLC compiler: struct {functionPointer m_func_address; // is 64-bit int m_vtable_index; int m_delta; int m_vtordisp;}; // gnu g The method of using a wire is used to optimize the Struct {UNION {FunctionPointer m_func_address; / / The value is always 4 multiple int m_vtable_index_2; // The value is divided by 2, the result is always odd}; int m_delta;}; for almost All compilers, Delta and Vindex are used to adjust the THIS pointer passed to the function, such as the Borland's calculation method is: Delta and Vindex are used to adjust the THIS pointer passed to the function, such as the calculation method of Borland is: AdjustedThis = * (THIS vindex -1) delta // If vindex! = 0adjustedthis = this delta // If VINDEX = 0 (where "*" is to extract the value in this address, AdjustedThis is the adjusted THIS pointer - translator's note ) * "Yes to extract the value in this address, AdjustedThis is the adjusted THIS pointer - Translator Note) Borland uses an optimization method: If this class is single inherited, the compiler will know that DELTA and VINDEX are the value of Delta and VINDEX. 0, so it can skip the above calculation method. The GNU compiler uses a strange optimization method.

It can be clearly seen that for multiple inheritance, you must view the VTABLE to get the Voffset (virtual function offset address) to calculate the THIS pointer. When you do these things, you may also save the function pointer in the VTABLE. Through these work, the compiler combines m_func_address and m_vtable_index to one (ie, in one Union), the compiler distinguishes the two variables of these two variables to make the value of the function pointer (m_func_address) to two after the result is an even number, and The virtual function table index (m_vtable_index_2) is odd in divided by 2 results. Their calculation method is: adjustedthis = this deltaif (Funcadr & 1) // If it is an odd call (* (* Delta (VINDEX 1) / 2) 4) Else // If it is an even Call Funcadr (where, Funcadr is the result of the function address divided by 2. - Translator Note) (where funcadr is the result of the function address divided by 2 results. - Translator Note) Inter-ITANIUM compiler (but not their x86 The compiler also uses the unknown_inheritance structure for virtual inheritance, so a virtual inheritance pointer has 20 bytes of size instead of 16 bytes of imagination. // itanium, unknown, and Virtual inheritance. Strus {functionPointer m_func_address; // is 64-bit int m_delta; int m_vtable_index; int m_vtable_index; int m_vtable,}; I can't guarantee that Comeau C is using the same technology as GNU It also cannot guarantee whether they use short instead of INT to narrow the size of this virtual function pointer to 8 bytes. The recently released Comeau C version uses Microsoft's compiler keyword (I think it is just ignoring these keywords without ignoring these keywords). Comeau C uses the same technique as the GNU, nor does it guarantee whether they use short instead of INT to narrow the size of this virtual function pointer to 8 bytes. The recently released Comeau C version uses Microsoft's compiler keyword (I think it is just ignoring these keywords without ignoring these keywords). The Digital Mars compiler (ie, the initial Zortech C to Symantec C ) uses a different optimization method. For a single inheritance class, a member function pointer is only the address of this function. However, when more complex inherits, this member function pointer points to a form conversion function (Thunk function), this function can implement the THUNK FUNCTION), this function can implement the necessary adjustments to the THIS pointer and can be used to call the actual member function . Whenever a multi-inheritance is involved, each member function has such a formal conversion function, which is very effective for the function call. But this means that when using multiple inheritance, the members of the subclass will not work to the base class member function pointer. It can be seen that such a compiler requires more compiled code than other compilers. Many compilers of embedded systems do not allow multiple inheritance.

In this way, these compilers avoid problems that may occur: A member function pointer is a normal function pointer with hidden THIS pointer parameters. Microsoft's "Smallest for Class" method Microsoft's compiler uses the optimization method similar to Borland. They all have optimal efficiency in the case of single inheritance. But unlike Borland, Microsoft omitted a pointer entry (entry) having a value of 0 under the default condition, I said this technology is "Smallest for Class" method: a single inheritance class, a member function pointer Only the address of the function is saved (m_func_address), so it has 4-byte long. For multiple inheritance classes, it has 8-bytes long since the offset address (M_Delta) is used. 12 bytes of virtual inheritance will be used. This method does save space, but there are other problems. First, conversion between a member function pointer between the subclass and the base class changes the size of the pointer! Therefore, the information will be lost. Second, when a member function pointer declares before its class definition, the compiler must calculate how many spaces to be assigned to this pointer, but doing so is unsafe, because the compiler cannot know the inheritance of this class before defining the way. For Intel C and early Microsoft compilers, the compiler speculates only the size of the pointer. Once you guess the error in the source file, your program will inexplicably crash when running. Therefore, some reserved words have been added to Microsoft's compiler: __ kindle_inheritance, and __multiple_inheritance, and __virtual_inheritance, and add some compiler switches, such as / vmg, so that all member function pointers have the same size, but The empty part of the original header member function pointer is filled with 0. The Borland compiler also added some compiler switches, but did not add new keywords. Intel's compiler identifies those keywords added to Microsoft, but it does not process these keywords in situations where you can find classes. For MSVC, where is the compiler you need to know where the class is ", there will be a THISP offset (VTORDISP), this value is constant for all the members in this class, but for each A class will be different. For MSVC, the adjusted THIS pointer is calculated by the original THIS pointer: Many embedded systems are not allowed to inherit. In this way, these compilers avoid problems that may occur: A member function pointer is a normal function pointer with hidden THIS pointer parameters. Microsoft's "Smallest for Class" method Microsoft's compiler uses the optimization method similar to Borland. They all have optimal efficiency in the case of single inheritance. But unlike Borland, Microsoft omitted a pointer entry (entry) having a value of 0 under the default condition, I said this technology is "Smallest for Class" method: a single inheritance class, a member function pointer Only the address of the function is saved (m_func_address), so it has 4-byte long. For multiple inheritance classes, it has 8-bytes long since the offset address (M_Delta) is used. 12 bytes of virtual inheritance will be used. This method does save space, but there are other problems.

First, conversion between a member function pointer between the subclass and the base class changes the size of the pointer! Therefore, the information will be lost. Second, when a member function pointer declares before its class definition, the compiler must calculate how many spaces to be assigned to this pointer, but doing so is unsafe, because the compiler cannot know the inheritance of this class before defining the way. For Intel C and early Microsoft compilers, the compiler speculates only the size of the pointer. Once you guess the error in the source file, your program will inexplicably crash when running. Therefore, some reserved words have been added to Microsoft's compiler: __ kindle_inheritance, and __multiple_inheritance, and __virtual_inheritance, and add some compiler switches, such as / vmg, so that all member function pointers have the same size, but The empty part of the original header member function pointer is filled with 0. The Borland compiler also added some compiler switches, but did not add new keywords. Intel's compiler identifies those keywords added to Microsoft, but it does not process these keywords in situations where you can find classes. For MSVC, where is the compiler you need to know where the class is ", there will be a THISP offset (VTORDISP), this value is constant for all the members in this class, but for each A class will be different. For MSVC, the adjusted THIS pointer is based on the original THIS pointer: Microsoft's "Smallest for Class" method of Microsoft's compiler uses the optimization method similar to the Borland. They all have optimal efficiency in the case of single inheritance. But unlike Borland, Microsoft omitted a pointer entry (entry) having a value of 0 under the default condition, I said this technology is "Smallest for Class" method: a single inheritance class, a member function pointer Only the address of the function is saved (m_func_address), so it has 4-byte long. For multiple inheritance classes, it has 8-bytes long since the offset address (M_Delta) is used. 12 bytes of virtual inheritance will be used. This method does save space, but there are other problems. First, conversion between a member function pointer between the subclass and the base class changes the size of the pointer! Therefore, the information will be lost. Second, when a member function pointer declares before its class definition, the compiler must calculate how many spaces to be assigned to this pointer, but doing so is unsafe, because the compiler cannot know the inheritance of this class before defining the way. For Intel C and early Microsoft compilers, the compiler speculates only the size of the pointer. Once you guess the error in the source file, your program will inexplicably crash when running. Therefore, some reserved words have been added to Microsoft's compiler: __ kindle_inheritance, and __multiple_inheritance, and __virtual_inheritance, and add some compiler switches, such as / vmg, so that all member function pointers have the same size, but The empty part of the original header member function pointer is filled with 0. The Borland compiler also added some compiler switches, but did not add new keywords.

Intel's compiler identifies those keywords added to Microsoft, but it does not process these keywords in situations where you can find classes. For MSVC, where is the compiler you need to know where the class is ", there will be a THISP offset (VTORDISP), this value is constant for all the members in this class, but for each A class will be different. For MSVC, the adjusted THIS pointer is based on the original THIS pointer: Microsoft's compiler uses the optimization method similar to the Borland. They all have optimal efficiency in the case of single inheritance. But unlike Borland, Microsoft omitted a pointer entry (entry) having a value of 0 under the default condition, I said this technology is "Smallest for Class" method: a single inheritance class, a member function pointer Only the address of the function is saved (m_func_address), so it has 4-byte long. For multiple inheritance classes, it has 8-bytes long since the offset address (M_Delta) is used. 12 bytes of virtual inheritance will be used. This method does save space, but there are other problems. First, conversion between a member function pointer between the subclass and the base class changes the size of the pointer! Therefore, the information will be lost. Second, when a member function pointer declares before its class definition, the compiler must calculate how many spaces to be assigned to this pointer, but doing so is unsafe, because the compiler cannot know the inheritance of this class before defining the way. For Intel C and early Microsoft compilers, the compiler speculates only the size of the pointer. Once you guess the error in the source file, your program will inexplicably crash when running. Therefore, some reserved words have been added to Microsoft's compiler: __ kindle_inheritance, and __multiple_inheritance, and __virtual_inheritance, and add some compiler switches, such as / vmg, so that all member function pointers have the same size, but The empty part of the original header member function pointer is filled with 0. The Borland compiler also added some compiler switches, but did not add new keywords. Intel's compiler identifies those keywords added to Microsoft, but it does not process these keywords in situations where you can find classes. For MSVC, where is the compiler you need to know where the class is ", there will be a THISP offset (VTORDISP), this value is constant for all the members in this class, but for each A class will be different. For MSVC, the adjusted THIS pointer is calculated based on the original THIS pointer: First, convert a member function pointer to the subclass and the base class change the size of the pointer! Therefore, the information will be lost. Second, when a member function pointer declares before its class definition, the compiler must calculate how many spaces to be assigned to this pointer, but doing so is unsafe, because the compiler cannot know the inheritance of this class before defining the way. For Intel C and early Microsoft compilers, the compiler speculates only the size of the pointer. Once you guess the error in the source file, your program will inexplicably crash when running.

Therefore, some reserved words have been added to Microsoft's compiler: __ kindle_inheritance, and __multiple_inheritance, and __virtual_inheritance, and add some compiler switches, such as / vmg, so that all member function pointers have the same size, but The empty part of the original header member function pointer is filled with 0. The Borland compiler also added some compiler switches, but did not add new keywords. Intel's compiler identifies those keywords added to Microsoft, but it does not process these keywords in situations where you can find classes. For MSVC, where is the compiler you need to know where the class is ", there will be a THISP offset (VTORDISP), this value is constant for all the members in this class, but for each A class will be different. For MSVC, the adjusted THIS pointer is based on the original THIS pointer, after the calculation of the following: For MSVC, the compiler needs to know where the class of vtable; usually there will be an offset of the THIS pointer (VTORDISP), this value is constant for member functions in all this class, but it will be different for each class. For MSVC, adjusted THIS pointer is based on the original THIS pointer: if (vindex = 0) // If it is not a virtual inheritance (_Virtual_inheritance) AdjustedThis = this deltaelse // If it is adjustedThis = THIS DELTA VTORDISP * (* (THIS VTORDISP) VINDEX) In the case of virtual inheritance, the value of VTORDISP does not save in the __virtual_inheritance pointer, but when the code is discovered, the compiler will Its corresponding assembly code is "embedded" in. However, for the inheritance of unknown types, the compiler needs to determine its inheritance type as much as possible, so the compiler divides the virtual inheritance Pointer into two types (__virtual_inheritance and __unknown_inheritance). In theory, all compiler designers should change and break through the implementation of the MFP (member function pointer). But in fact, this is not good, because this makes it possible to change the current number of code written. Microsoft has published a very old article (http://msdn.microsoft.com/archive/en-us/dnarvc/html/jangrayHood.asp) to explain the implementation details of Visual C operation. This article is written by Jan Gray, and he has designed Microsoft C object model in 1990. Although this article was published in 1994, this article is still very important - this means that the C object model is not changed in 15 years (1990 to 2004). Now, I think you know too much about the members' function pointer. What is the point? I have established a rule for you. Although various compilers have a lot of implementation methods in this regard, there are also useful common: no matter which form of class, the assembly language code generated by calling a member function pointer is exactly the same.

There is a special case to use the non-standard compiler of "Smallest for Class" technology, even in this case, the difference is also very small. This fact allows us to continue to explore how to build high-performance delegates. (Send) The value of VTORDISP does not save in the __virtual_inheritance pointer, but when the code is discovered, the compiler "is" embedded "in the compiler. However, for the inheritance of unknown types, the compiler needs to determine its inheritance type as much as possible, so the compiler divides the virtual inheritance Pointer into two types (__virtual_inheritance and __unknown_inheritance). In theory, all compiler designers should change and break through the implementation of the MFP (member function pointer). But in fact, this is not good, because this makes it possible to change the current number of code written. Microsoft has published a very old article (http://msdn.microsoft.com/archive/en-us/dnarvc/html/jangrayHood.asp) to explain the implementation details of Visual C operation. This article is written by Jan Gray, and he has designed Microsoft C object model in 1990. Although this article was published in 1994, this article is still very important - this means that the C object model is not changed in 15 years (1990 to 2004). Now, I think you know too much about the members' function pointer. What is the point? I have established a rule for you. Although various compilers have a lot of implementation methods in this regard, there are also useful common: no matter which form of class, the assembly language code generated by calling a member function pointer is exactly the same. There is a special case to use the non-standard compiler of "Smallest for Class" technology, even in this case, the difference is also very small. This fact allows us to continue to explore how to build high-performance delegates. (Continued) The implementation of the MFP (member function pointer) has changed and breakthrough. But in fact, this is not good, because this makes it possible to change the current number of code written. Microsoft has published a very old article (http://msdn.microsoft.com/archive/en-us/dnarvc/html/jangrayHood.asp) to explain the implementation details of Visual C operation. This article is written by Jan Gray, and he has designed Microsoft C object model in 1990. Although this article was published in 1994, this article is still very important - this means that the C object model is not changed in 15 years (1990 to 2004). Now, I think you know too much about the members' function pointer. What is the point? I have established a rule for you. Although various compilers have a lot of implementation methods in this regard, there are also useful common: no matter which form of class, the assembly language code generated by calling a member function pointer is exactly the same. There is a special case to use the non-standard compiler of "Smallest for Class" technology, even in this case, the difference is also very small. This fact allows us to continue to explore how to build high-performance delegates.

转载请注明原文地址:https://www.9cbs.com/read-36424.html

New Post(0)