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

xiaoxiao2021-03-05  24

MEMBER FUNCTION POINTERS AND THE FASTEST POSSIBLE C Delegates

Written: Don Clugston

Don Clugston

Translation: Zhou Xiang

Primer

There is no real object-oriented function pointer in standard C . This is unfortunate for C because it has proved its valuable value in some languages ​​because of an object-oriented pointer (also called "closure" or "delegate"). In Delphi (Object Pascal), object-oriented function pointers are the basis of the Borland visualization set library (VCL, Visual Component Library). At present, C # makes the concept of "delegate" becomes more popular, which is also showing the success of C # this language. In many applications, "delegate" simplifies the design pattern of loosely coupled objects [GOF]. This feature undoubtedly produces a lot of role in standard C . Unfortunately, there is no "delegate" in C , which only provides member function pointers. Many programmers have never used a function pointer, which is reasonable. Because the function pointer itself has a lot of strange syntax rules (such as "Member Function Pointers). Many programmers have never used function pointers, which is a specific reason. Because the function pointer itself has a lot of strange syntax rules (for example" - > * "And". * "Operator), and it is difficult to find their accurate meaning, and you will find a better way to avoid using function pointers. More ironic is: in fact, compilers writing If you implement the "delegate", you will be more easy than him. In this article, I want to uncover the "mysterious cover" of the member function pointer. Syntax of the membership function pointer After the characteristics, I will explain the member function pointer to the reader in some common compilers, and then I will show you how to effectively achieve "commission". Finally, I will use these deep knowledge to you. Showing techniques for implementing optimization and reliable implementation on the C compiler. For example, in Visual C (6.0, .Net, And .NET 2003) calls, the compiler is only Generate two lines of assembly code!

C

There is no real object-oriented function pointer. This

C

It is unfortunate because of an object-oriented pointer (also called "closure (

Closure

) "Or" commission ("

Delegate

")" In some languages ​​have proven its valuable value.

Delphi (Object Pascal)

In the object-oriented function pointer is

Borland

Visualization set library

VCL

,

Visual Component Library

)Foundation. At present,

C #

Let the concept of "entrusted" becomes more popular, which is also shown

C #

This language is successful. In many applications, "delegate" simplifies the design pattern of loosely coupled objects.

[Gof]

. This feature is undoubtedly in the standard

C

It will also have a lot of role. Unfortunately, there is no "delegate" in C , which only provides member function pointers. Many programmers have never used a function pointer, which is reasonable. Because the function pointer itself has a lot of strange syntax rules (such as "Member Function Pointers). Many programmers have never used function pointers, which is a specific reason. Because the function pointer itself has a lot of strange syntax rules (for example" - > * "And". * "Operator), and it is difficult to find their accurate meaning, and you will find a better way to avoid using function pointers. More ironic is: in fact, compilers writing If you implement the "delegate", you will be more easy than him. In this article, I want to uncover the "mysterious cover" of the member function pointer. Syntax of the membership function pointer After the characteristics, I will explain the member function pointer to the reader in some common compilers, and then I will show you how to effectively achieve "commission". Finally, I will use these deep knowledge to you. Showing techniques for implementing optimization and reliable implementation on the C compiler. For example, in Visual C (6.0, .Net, And .NET 2003) calls, the compiler is only Generate two lines of assembly code! C

There is no "delegate", which only provides member function pointers (MEMBER FUNCTION POINTERS). Many programmers have never used a function pointer, which is reasonable. Because the function pointer itself has a lot of strange grammar rules (such as "

MEMBER FUNCTION POINTERS

). Many programmers have never used a function pointer, which is reasonable. Because the function pointer itself has a lot of strange grammar rules (such as "

-> *

"with"

. *

"Operator), and it is difficult to find their accurate meaning, and you will find a better way to avoid using function pointers. More ironic is: in fact, if the compiler is writer, if the compiler will be" delegated " More than he is more expensive! In this article, I will uncover the "mysterious cover" of the member function pointer. After the grammatical and characteristics of the member function pointer, I will The reader explains how the member function pointer is implemented in some common compilers, and then I will show you how to effectively achieve "commission". Finally, I will use these exquisite knowledge to show you on the C compiler. Optimized and reliable technology. For example, in Visual C (6.0, .NET, And .NET 2003) calls, the compiler only generates two line assembly code!

C

Improved and reliable technology on the compiler. For example,

Visual C (6.0, .NET, AND .NET 2003)

A single target commission (

SINGLE-TARGET DELEGATE

The call, the compiler only generates two line assembly code!

Function pointer

Let's review the function pointer. In C and C languages, a function pointer named my_func_ptr pointing to a function that is parameter with an int and a char *, which returns a floating point value, as follows:

C

with

C

In the language, a name is named

MY_FUNC_PTR function pointer points to one with one

int

and a

Char *

For the function of the parameter, this function returns a floating point value, as follows:

FLOAT (* my_func_ptr) (int, char *); // I strongly recommend you to use the typedef keyword for your convenience.

For the sake of understanding, I highly recommend you to use

Typedef

Keyword.

// If this is not this, when the function pointer is passed as a function of the parameter,

If this is not this, when the function pointer is passed as a function of a function,

// The program will become obscured.

The program will become obscured.

// This kind, the statement should be as follows:

In this case, the statement should be as follows:

Typedef float (int, char *); myfuncptype my_func_ptr;

It should be noted that the type of function pointer to each function should be different. In Microsoft Visual C (hereinafter referred to as MSVC), there are different types of different call modes: __ cdecl, __stdcall, and __fastcall. If your function pointer points to a function of a Float Some_Func (int, char *), you can do this:

Microsoft Visual C

(Hereinafter

MSVC

), There are different types of different call modes:

__cdecl, __stdcall,

with

__fastcall

. If your function pointer points to a type

Float Some_func (int, char *)

The function can do this:

MY_FUNC_PTR = Some_Func;

When you want to call the function you point to, you can write:

(* my_func_ptr) (7, "Arbitrary String");

You can convert a type of function pointer into another function pointer type, but you can't point a function pointer to a Void * type data pointer. Other conversion operations are not detailed. A function pointer can be set to 0 to indicate that it is an empty pointer. All comparison operators (==,! =, <,>, <=,> =) Can be used, can use "== 0" or pass a explicit Boolean conversion to test the pointer is empty (NULL) . In the C language, the function pointer is usually used as a parameter like QSORT, or as a callback function of the Windows system function. There are many other applications in the function pointer. The implementation of the function pointer is simple: they are just "code pointer", which is reflected in the assembly language is the first address used to hold the subroutine code. The presence of this function pointer is just to ensure the correct invoking specification.

Void *

Type data pointer. Other conversion operations are not detailed. A function pointer can be set to

0

To indicate that it is an empty pointer. All comparison operators (

==,! =, <,>, <=,> =

) Can be used, you can use "

== 0

"Or through an explicit Boolean conversion to test whether the pointer is empty (

NULL

). In the C language, the function pointer is usually used as a parameter like QSORT, or as a callback function of the Windows system function. There are many other applications in the function pointer. The implementation of the function pointer is simple: they are just "code pointer", which is reflected in the assembly language is the first address used to hold the subroutine code. The presence of this function pointer is just to ensure the correct invoking specification. C

In the language, the function pointer is usually used

Qsort

The same will use the function as a parameter, or as

Windows

The callback function of the system function, and the like. There are many other applications in the function pointer. The implementation of the function pointer is simple: they are just "code pointers (

Code Pointer

) ", They are embodied in assembly language is the first address used to hold the subroutine code. The existence of this function pointer is just to ensure the correct call specification.

Member function pointer

In a C program, many functions are members functions, that is, these functions are part of a certain class. You can't point to a member function like a normal function pointer, the correct approach should be, you must use a member function pointer. The pointer to a member function points to a member function in the class and has the same parameters as before, and the declaration is as follows:

C

Many functions are a member function, that is, these functions are part of a certain class. You can't point to a member function like a normal function pointer, the correct approach should be, you must use a member function pointer. The pointer to a member function points to a member function in the class and has the same parameters as before, and the declaration is as follows:

FLOAT (SomeClass :: * my_Memfunc_ptr) (int, char *);

/ / Declare as follows: for member functions modified using const keywords:

FLOAT (SomeClass :: * my_const_memfunc_ptr) (int, char *) const;

Note The special operator (:: *) is used, and "someclass" is part of the declaration. Member function pointer has a terrible restriction: they can only point to member functions in a particular class. For each parameter, there is a need for different member function pointer types, and there are different function pointer types for each function and different classes using const modified functions and different classes. In MSVC, there is a different call type for the following four calls: __ cdecl, __stddcall, __fastcall, and __thiscall. (_Thiscall is the default way, interesting is that there is no detailed description of the __thiscall keyword in any official document, but it often appears in the error message. If you explicitly use it, you will see " It is retained an error prompt for IT IS Reserved for Future Use.) If you use a member function pointer, you'd better use TypeDef to prevent confusion. Write the function pointer to the function of Float SomeClass :: Some_member_func (int, char *), you can write:

:: *

),and"

Someclass

"It is part of the statement. The member function pointer has a terrible restriction: they can only point to member functions in a particular class. For each parameter, there is a need for different member function pointer types, and for each use

Const

Modified functions and functions in different classes, also have different function pointer types. in

MSVC

Among the four modes of calls, there is a different call type:

__CDECL, __STDCALL, __FASTCALL, and

__thiscall

. (

__thiscall

It is the default way, interesting is that there is no pair in any official document

__thiscall

Detailed description of keywords, but it often appears in the error message. If you explicitly use it, you will see "It is reserved as later use (

IT is Reserved for Future Use

The error message.) If you use the member function pointer, you should use it.

Typedef

To prevent confusion. Write the function pointer to the function of Float SomeClass :: Some_member_func (int, char *), you can write:

Float SomeClass :: Some_member_func (int, char *)

Function, you can write this:

MY_MEMFUNC_PTR = & SomeClass :: Some_member_func;

Many compilers (such as MSVC) will let you go to "&", while others (such as GNU G ) need to add "&", so I suggest add it to it when handwriting. To call a member function pointer, you need to build an instance of SomeClass and use special operators "-> *", which is lower than the priority of this operator, you need to put it in parentheses.

MSVC

) Will let you go "

&

", And some other compilers (such as

GNU G

) Need to be added "

&

", So I suggest add it to the member function pointer when you write it. To call the member function pointer, you need to build

Someclass

One example, and use special operators

-> *

", The priority of this operator is low, you need to put it appropriately into parentheses.

Someclass * x = new someclass;

(x -> * my_memfunc_ptr) (6, "Another Arbitrary Parameter);

// If the class is on the stack, you can also use the ". *" Operator.

SomeClass Y;

(y. * my_memfunc_ptr) (15, "DiffERENT Parameters this Time);

Don't blame me for using such a strange syntax - looks that C designers have a sincere feelings for punctuation! C increases three special operators to support member pointers relative to C. ":: *" is used to declare the pointer, and "-> *" and ". *" Are used to call the functionality points to the pointer. It looks that the excessive concern of some of the part of a language blurred and fewer use is excessive. (You can of course overload "-> *" these operators, but this is not the range to be involved in this article.) A member function pointer can be set to 0, and you can use "==" and "! =" Comparison operator However, this can only be limited between the pointers of the member functions in the same class. Any member function pointer can be compared to 0 to determine if it is empty. Unlike function pointers, unequal operators (<,>, <=,> =) are not available to member function pointers.

C

Designers have a sincere feelings for punctuation!

C

Relative to

C

Increase three special operators to support member pointers. "

:: *

"Declaration for pointers, and"

-> *

"with"

. *

"Used to call the function to point to the pointer. This looks too much attention to the part of a language blurred and rarely used is redundant. (Of course you can overload" -> *

"These operators, but this is not the range to be involved in this article.) A member function pointer can be set to 0, and can use" == "and"! = "Comparison operators, but can only be limited to the same class This comparison is made between the pointer of the member function. Any member function pointer can be compared to 0 to determine if it is empty. Different from the function pointer, unequal operators (<,>, <=,> =) pair Member function pointer is not available.

0

And you can use "

==

"with"

! =

"Compare operators, but can only be limited between the pointers of the member functions in the same class. Any member function pointer can be

0

Do comparison to determine if it is empty. Unlike function pointers, unequal operators (

<,>, <=,> =

) Is not available to the member function pointer.

Member function pointer weird

Member function pointer sometimes is very strange. First, you can't use a member function to point to a static member function, you must use a normal function pointer ("Member Function Pointer", which should actually be "non-static member function pointer" Correct). Second, when using the inheritance of the class, there will be some strange situations. For example, the following code will compile success in MSVC (note code notes):

MSVC

Successfully compiled (note code notes):

#include "stdio.h" Class SomeClass {public: Virtual Void Some_member_func (int X, char * p) {Printf ("in someclass");};}; class deivedclass: public someclass {public: // If you The annotation of a row, the line with line (*) will have an error

If you sell the annotation of the next line, bring

line (*)

The one will have an error

// virtual void Some_member_func (int x, char * p) {printf ("in derivedclass");};}; int main () {// statement SomeClass member function pointer

statement

Someclass

Member function pointer

Typedef void (INT, CHAR *); SomeClassmfp my_memfunc_ptr; my_memfunc_ptr = & derivedclass :: some_member_func; // ---- line (*) returnograph;

} Weird is, & derivedclass :: some_member_func is a member function pointer for a SomeClass class, not a member function pointer of the DeriveDclass class! (Some compilers are slightly different: For example, for Digital Mars C , in the above example, & derivedClass :: some_member_func will be considered unfained.) However, if you override (override) in the DeriveDClass class, the code is Unable to compile because the current & derivedClass :: Some_member_func has become a member function pointer in the DeriveDClass class! The type conversion between member function pointers is a very vague topic discussed. During the standardization of C , when the member function pointer involving the inherited class, for a member function pointer that converts the member function pointer to the base class or whether a member of the subclass member function pointer can be used and a class member The function pointer translates into another member function of the unrelated class, and people have had a very intense debate. Unfortunately, the different compiler producers have implemented their compilers according to their different answers to these issues before they make a decision. Based on standards (Section 5.2.2.2.2/9), you can use ReinterPret_CAST to save a member function of classes that are not related to the category of categories in a member function pointer. The final result of the issue of the member function pointer conversion is not determined. What you can do now is like previously - convert the member function pointer into a pointer to the member function of this class. Behind the article I will continue to discuss this problem, because this is a topic that each compiler has not reached a consensus on such a standard. In some compilers, there is often a strange thing between the transition between the base class and subclauses. When involving multiple inheritance, when using ReinterPret_CAST converts subclasses into base classes, it is possible to compile a particular compiler, and may also be compiled, depending on the list of base classs in the subclass. The order of the base class! Here is an example: & derivedclass :: some_member_func

Is an

Someclass

Class member function pointer, not

DerivedClass

Class member function pointer! (Some compilers are slightly different: for example, for

DIGITAL MARS C

In the above example,

& Derivedclass :: Some_member_func

It will be considered not defined. ), If

DerivedClass

Override in the class (

Override

) The

Some_member_func

Function, code cannot be compiled, because now

& Derivedclass :: Some_member_func

has become

DerivedClass

Member function pointer in the class! The type conversion between member function pointers is a very vague topic discussed. During the standardization of C , when the member function pointer involving the inherited class, for a member function pointer that converts the member function pointer to the base class or whether a member of the subclass member function pointer can be used and a class member The function pointer translates into another member function of the unrelated class, and people have had a very intense debate. Unfortunately, the different compiler producers have implemented their compilers according to their different answers to these issues before they make a decision. Based on standards (Section 5.2.2.2.2/9), you can use ReinterPret_CAST to save a member function of classes that are not related to the category of categories in a member function pointer. The final result of the issue of the member function pointer conversion is not determined. What you can do now is like previously - convert the member function pointer into a pointer to the member function of this class. Behind the article I will continue to discuss this problem, because this is a topic that each compiler has not reached a consensus on such a standard. In some compilers, there is often a strange thing between the transition between the base class and subclauses. When involving multiple inheritance, when using ReinterPret_CAST converts subclasses into base classes, it is possible to compile a particular compiler, and may also be compiled, depending on the list of base classs in the subclass. The order of the base class! Here is an example: C

During the standardization process, when a member function pointer involving inherited classes, for a member function pointer that converts member function pointers to a base class or whether a class member function pointer can be used for a class member function? The problem of transforming into another member of the unrelated class, people have had a fierce argument. Unfortunately, the different compiler producers have implemented their compilers according to their different answers to these issues before they make a decision. According to standards

5.2.10 / 9

Festival), you can use

Reinterpret_cast

Save a member function of a class that is not related to the original class in a member function pointer. The final result of the issue of the member function pointer conversion is not determined. What you can do now is like previously - convert the member function pointer into a pointer to the member function of this class. Behind the article I will continue to discuss this problem, because this is a topic that each compiler has not reached a consensus on such a standard. In some compilers, there is often a strange thing between the transition between the base class and subclauses. When involving multiple inheritance, when using ReinterPret_CAST converts subclasses into base classes, it is possible to compile a particular compiler, and may also be compiled, depending on the list of base classs in the subclass. The order of the base class! Here is an example:

Reinterpret_cast

When the subclass is converted into a base class, it is possible to compile a particular compiler, and it may be compiled, depending on the order in the base class in the sub-list list! Here is an example:

Class Derived: Public Base1, Public Base2 // Sample (A) Class Derived2: Public Base2, Public Base1 // Sample (B) Typedef Void (Derived :: * Derived_mfp) (); typedef void (derived2 :: * derived2_mfp) ( ); typedef void (base1 :: * base1mfp) (); typedef void (base2 :: * base2mfp) (); Derived_mfp x;

Happening

(a) Class Derived2: Public Base2, Public Base1 // Sample (b) typef void (Derived :: * Derived_mfp) (); typedef void (derived2 :: * derived2_mfp) (); typedef void (base1 :: * base1mfp) (); typedef void (base2 :: * base2mfp) (); derived_mfp x; situation

(b) typef void (Derived :: * Derived_mfp) (); typedef void (Derived2 :: * Derived2_mfp) (); typedef void (base1 :: * base1mfp) (); typedef void (base2 :: * base2mfp) () Derived_mfp x;

For the situation (a), static_cast

(x) is legal, but static_cast

(x) is wrong. However, the situation (b) (b) is opposite to it. You can only transfer the members of the subclass into the member function pointer of the first base class! If you have to experiment, MSVC will issue a Warning C4407, while Digital Mars C will compile errors. If you replace Static_cast_cast, both compilers have errors, but both compilers have different reasons. But some compilers are ignored by this detail, everyone can be careful! The other interesting rules in standard C are: You can declare its member function pointer before class definition. This will have some compiler that there will be some unpredictable side effects. I will discuss this problem later, now you know how to avoid this as much as possible. It is worth noting that the Member Data Pointer is also provided like a member function pointer, standard C . They have the same operator, and some implementation principles are also the same. They are used in some implementations in Stl :: Stable_Sort, and I will no longer mention many other applications. Both compilers have errors, but both compilers have different reasons. But some compilers are ignored by this detail, everyone can be careful!

(a), static_cast

(x)

It is legal, and

STATIC_CAST

(x)

It is wrong. However, the situation (b)

(b)

It is opposite to it. You can only transfer the members of the subclass into the member function pointer of the first base class! If you have to experiment,

MSVC

Put out

C4407

Number warning, and

DIGITAL MARS C

Compile errors will occur. If used

Reinterpret_cast

instead

Static_cast, both compilers have errors, but both compilers have different reasons. But some compilers are ignored by this detail, everyone can be careful! The other interesting rules in standard C are: You can declare its member function pointer before class definition. This will have some compiler that there will be some unpredictable side effects. I will discuss this problem later, now you know how to avoid this as much as possible. It is worth noting that the Member Data Pointer is also provided like a member function pointer, standard C . They have the same operator, and some implementation principles are also the same. They are used in some implementations in Stl :: Stable_Sort, and I will no longer mention many other applications. Both compilers have errors, but both compilers have different reasons. But some compilers are ignored by this detail, everyone can be careful! The other interesting rules in standard C are: You can declare its member function pointer before class definition. This will have some compiler that there will be some unpredictable side effects. I will discuss this problem later, now you know how to avoid this as much as possible.

It is worth noting that the Member Data Pointer is also provided like a member function pointer, standard C . They have the same operator, and some implementation principles are also the same. They are used in some implementations in Stl :: Stable_Sort, and I will no longer mention many other applications.

It is worth noting that the Member Data Pointer is also provided like a member function pointer, standard C . They have the same operator, and some implementation principles are also the same. They are used in some implementations in Stl :: Stable_Sort, and I will no longer mention many other applications.

Member function pointer

Now you may think that the member function pointer is a bit of strange. But what can it be used? I have made a very broad survey on this online. Finally, I summarize two reasons for the use of member function pointers:

Used to do an example to see the C beginners, help them learn grammar; or to achieve "delegate"! The use of the member function pointer in the single-line function adaptor in the STL and Boost libraries is negligible and allows you to use the member function and the standard algorithm. But their most important applications are in different types of application frameworks, such as their core of the MFC message system. When you use the MFC message mapping macro (such as on_command), you will assemble a sequence containing a message ID and a member function pointer (type: ccmdtarget :: * member function pointer). This is one of the reasons why the MFC class must inherit ccmdTarget can handle the message. However, a variety of different message processing functions have different parameter lists (such as the type CDC * of the first parameter of the onDRAW processing function, so the sequence must contain various types of member function pointers. How does the MFC do this? MFC uses a terrible compiler vulnerability (HACK) that puts all possible members function pointers in a huge union (Union), avoiding a C type matching check that is often required. (Look at it (see union of AFXImpl.h and cmdtarg.cpp, you will find this horror fact.) Because MFC has such an important part of code, the truth is that all compilers are This vulnerability has opened a green light. (However, we will see later, if some classes use multiple inheritance, this vulnerability will not work in the MSVC, which is only the reason for only single inheritance when using the MFC. .) There are similar vulnerabilities in Boost :: function (but not too serious). It looks like you want to do anything interesting to do anything about member function pointers, you must prepare for the vulnerability to this language. If you want to deny the C member function pointer design has a defective point of view, it seems difficult. I have something to indicate in this article: "Allow the conversion between member function pointers, and The function is not allowed to be called after the conversion is completed, "is included in this rule, and this rule is incorporated into C standard. First, many popular compilers do not support this conversion (so conversion is standard requirements) However, it is not portable). Second, all compilers, if the conversion is successful, when calling the converted member function pointer, you can still achieve your expected function: The compiler does not have a so-called "undefined behavior (undefined behavior "This type of error is necessary (invocation) is feasible, but this is not a standard!). Third, allow conversion without allowing calls to be completely useless, only conversion and calling are feasible, can be convenient Effectively achieve entrustment, so that this language benefits. In order to make you confident that this controversial argument, consider only the following code in a file, this code is legal: MSVC will not work, This is exactly the use of a single inheritage when using the MFC.) There are similar vulnerabilities (but not too serious) in Boost :: function.

It seems that if you want to do anything more interesting about the member function pointer, you must prepare for challenges with this language vulnerability. If you want to deny the member function of C , it is difficult to design a defective point of view. In this article, I have something to indicate: "Allow the member function pointer to convert (CAST) without allowing the function to be called after the conversion is complete", incorporating this rule, incorporating this rule into C The standard is coming. First, a lot of popular compilers do not support this conversion (so conversion is standard requirements, but not portable). Second, all the compilers, if the conversion is successful, you can still achieve your expected function when calling the converted member function pointer: The compiler does not have the necessary "undefined behavior". " (Invocation) is feasible, but this is not a standard!). Third, allowing conversion without allowing calls to be completely useless, only converts and calls are feasible, to facilitate entrustment, so that this language benefits. In order to make you confident that this controversial argument, consider only the following code in a file, this code is legal: in order to implement "delegate"! The use of the member function pointer in the single-line function adaptor in the STL and Boost libraries is negligible and allows you to use the member function and the standard algorithm. But their most important applications are in different types of application frameworks, such as their core of the MFC message system. When you use the MFC message mapping macro (such as on_command), you will assemble a sequence containing a message ID and a member function pointer (type: ccmdtarget :: * member function pointer). This is one of the reasons why the MFC class must inherit ccmdTarget can handle the message. However, a variety of different message processing functions have different parameter lists (such as the type CDC * of the first parameter of the onDRAW processing function, so the sequence must contain various types of member function pointers. How does the MFC do this? MFC uses a terrible compiler vulnerability (HACK) that puts all possible members function pointers in a huge union (Union), avoiding a C type matching check that is often required. (Look at it (see union of AFXImpl.h and cmdtarg.cpp, you will find this horror fact.) Because MFC has such an important part of code, the truth is that all compilers are This vulnerability has opened a green light. (However, we will see later, if some classes use multiple inheritance, this vulnerability will not work in the MSVC, which is only the reason for only single inheritance when using the MFC. .) There are similar vulnerabilities in Boost :: function (but not too serious). It looks like you want to do anything interesting to do anything about member function pointers, you must prepare for the vulnerability to this language. If you want to deny the C member function pointer design a defective point of view, it seems difficult.

In this article, I have something to indicate: "Allow the member function pointer to convert (CAST) without allowing the function to be called after the conversion is complete", incorporating this rule, incorporating this rule into C The standard is coming. First, a lot of popular compilers do not support this conversion (so conversion is standard requirements, but not portable). Second, all the compilers, if the conversion is successful, you can still achieve your expected function when calling the converted member function pointer: The compiler does not have the necessary "undefined behavior". " (Invocation) is feasible, but this is not a standard!). Third, allowing conversion without allowing calls to be completely useless, only converts and calls are feasible, to facilitate entrustment, so that this language benefits. In order for you to believe this controversial, consider only the following code in a file, this code is legal: MSVC will not work, this is only a single one when using MFC. The reason for inheritance. ) There are similar vulnerabilities (but not too serious) in Boost :: function. It seems that if you want to do anything more interesting about the member function pointer, you must prepare for challenges with this language vulnerability. If you want to deny the member function of C , it is difficult to design a defective point of view. In this article, I have something to indicate: "Allow the member function pointer to convert (CAST) without allowing the function to be called after the conversion is complete", incorporating this rule, incorporating this rule into C The standard is coming. First, a lot of popular compilers do not support this conversion (so conversion is standard requirements, but not portable). Second, all the compilers, if the conversion is successful, you can still achieve your expected function when calling the converted member function pointer: The compiler does not have the necessary "undefined behavior". " (Invocation) is feasible, but this is not a standard!). Third, allowing conversion without allowing calls to be completely useless, only converts and calls are feasible, to facilitate entrustment, so that this language benefits. In order to make you confident that this controversial argument, consider only the following code in a file, this code is legal: C

Beginners see, help them learn grammar; or to achieve "delegate"! The use of the member function pointer in the single-line function adaptor in the STL and Boost libraries is negligible and allows you to use the member function and the standard algorithm. But their most important applications are in different types of application frameworks, such as their core of the MFC message system. When you use the MFC message mapping macro (such as on_command), you will assemble a sequence containing a message ID and a member function pointer (type: ccmdtarget :: * member function pointer). This is one of the reasons why the MFC class must inherit ccmdTarget can handle the message. However, a variety of different message processing functions have different parameter lists (such as the type CDC * of the first parameter of the onDRAW processing function, so the sequence must contain various types of member function pointers. How does the MFC do this? MFC uses a terrible compiler vulnerability (HACK) that puts all possible members function pointers in a huge union (Union), avoiding a C type matching check that is often required. (Look at it (see union of AFXImpl.h and cmdtarg.cpp, you will find this horror fact.) Because MFC has such an important part of code, the truth is that all compilers are This vulnerability has opened a green light. (However, we will see later, if some classes use multiple inheritance, this vulnerability will not work in the MSVC, which is only the reason for only single inheritance when using the MFC. .) There are similar vulnerabilities in Boost :: function (but not too serious). It looks like you want to do anything interesting to do anything about member function pointers, you must prepare for the vulnerability to this language. If you want to deny the C member function pointer design has a defective point of view, it seems difficult. I have something to indicate in this article: "Allow the conversion between member function pointers, and The function is not allowed to be called after the conversion is completed, "is included in this rule, and this rule is incorporated into C standard. First, many popular compilers do not support this conversion (so conversion is standard requirements) However, it is not portable). Second, all compilers, if the conversion is successful, when calling the converted member function pointer, you can still achieve your expected function: The compiler does not have a so-called "undefined behavior (undefined behavior "This type of error is necessary (invocation) is feasible, but this is not a standard!). Third, allow conversion without allowing calls to be completely useless, only conversion and calling are feasible, can be convenient Effectively achieve entrustment, so that this language benefits. In order to make you confident that this controversial argument, consider only the following code in a file, this code is legal: MSVC will not work, This is exactly the use of a single inheritage when using the MFC.) There are similar vulnerabilities (but not too serious) in Boost :: function.

It seems that if you want to do anything more interesting about the member function pointer, you must prepare for challenges with this language vulnerability. If you want to deny the member function of C , it is difficult to design a defective point of view. In this article, I have something to indicate: "Allow the member function pointer to convert (CAST) without allowing the function to be called after the conversion is complete", incorporating this rule, incorporating this rule into C The standard is coming. First, a lot of popular compilers do not support this conversion (so conversion is standard requirements, but not portable). Second, all the compilers, if the conversion is successful, you can still achieve your expected function when calling the converted member function pointer: The compiler does not have the necessary "undefined behavior". " (Invocation) is feasible, but this is not a standard!). Third, allowing conversion without allowing calls to be completely useless, only converts and calls are feasible, to facilitate entrustment, so that this language benefits. In order to let you believe this controversial argument, consider only the following code in a file, this code is legal: delegate

) "! The use of the member function pointer in the single-line function adaptor in the single-line function adaptor, and allows you to mix the member function and the standard algorithm. But their most important applications are different In the type of application framework, for example, they form the core of the MFC message system. When you use the MFC message mapping macro (such as on_command), you will assemble a message ID and member function pointer (type: ccmdtarget :: * Sequence of member function pointers). This is one of the reasons why the MFC class must inherit ccmdTarget can handle the reason for the message. However, various different message processing functions have different parameter lists (such as the first parameter of the onDRAW handler). CDC *), so the sequence must contain a variety of different types of member function pointers. How does MFC do this? MFC uses a terrible compiler vulnerability (HACK), it will make all possible members function pointers Put it in a huge union (Union) to avoid the usual C type matching check. (Take a look (see union of AFXIMPL.H and cmdtarg.cpp "Union, you will find this terror The facts.) Because the MFC has such an important part of the code, the fact is that all the compilers have opened a green light for this vulnerability. (However, we will see later, if some classes used multiple inheritance, this vulnerability There is no role in the MSVC, which is only the reason why you can use only a single inheritage when using the MFC.) There are similar vulnerabilities (but not too serious) in Boost :: Function. It looks like you want to do anything Member function pointer is more interesting, you must prepare for challenge with this language vulnerability. If you want to negate the C member function pointer design defective point of view, it seems difficult. Write this In the article, I have something to indicate: "Allow the conversion between the member function pointer, not to call the function after the conversion is completed", incorporate this rule, incorporate this rule into C standard is ridiculous First, many popular compilers do not support this conversion (so conversion is standard requirements, but not portable). Second, all compilers, if the conversion is successful, call the converted member function pointer You can still achieve your expected function: The compiler does not have the necessary "undefined behavior" error (invocation is feasible, but this is not the standard!). Third Allow conversion without allowing calls to be completely useless, only converts and calls can be easily and effectively achieved, so that this language benefits . In order to make you confident that this controversial argument, consider only the following code in a file, this code is legal: MSVC

Will not work, this is being used

MFC

You can only use a single inheritance. ) There are similar vulnerabilities (but not too serious) in Boost :: function. It seems that if you want to do anything more interesting about the member function pointer, you must prepare for challenges with this language vulnerability. If you want to deny the member function of C , it is difficult to design a defective point of view. In this article, I have something to indicate: "Allow the member function pointer to convert (CAST) without allowing the function to be called after the conversion is complete", incorporating this rule, incorporating this rule into C The standard is coming. First, a lot of popular compilers do not support this conversion (so conversion is standard requirements, but not portable). Second, all the compilers, if the conversion is successful, you can still achieve your expected function when calling the converted member function pointer: The compiler does not have the necessary "undefined behavior". " (Invocation) is feasible, but this is not a standard!). Third, allowing conversion without allowing calls to be completely useless, only converts and calls are feasible, to facilitate entrustment, so that this language benefits. In order to make you confident that this is controversial, consider only the following code in one file, this code is legal: STL

with

Boost

Library single line function adapter (

One-Line Function Adaptor

The use is insignificant and allows you to use the member function and the standard algorithm. But their most important applications are in different types of application frameworks, such as they form.

MFC

The core of the message system. When you use the MFC message mapping macro (such as on_command), you will assemble a sequence containing a message ID and a member function pointer (type: ccmdtarget :: * member function pointer). This is one of the reasons why the MFC class must inherit ccmdTarget can handle the message. However, a variety of different message processing functions have different parameter lists (such as the type CDC * of the first parameter of the onDRAW processing function, so the sequence must contain various types of member function pointers. How does the MFC do this? MFC uses a terrible compiler vulnerability (HACK) that puts all possible members function pointers in a huge union (Union), avoiding a C type matching check that is often required. (Look at it (see union of AFXImpl.h and cmdtarg.cpp, you will find this horror fact.) Because MFC has such an important part of code, the truth is that all compilers are This vulnerability has opened a green light. (However, we will see later, if some classes use multiple inheritance, this vulnerability will not work in the MSVC, which is only the reason for only single inheritance when using the MFC. .) There are similar vulnerabilities in Boost :: function (but not too serious). It looks like you want to do anything interesting to do anything about member function pointers, you must prepare for the vulnerability to this language. If you want to deny the C member function pointer design has a defective point of view, it seems difficult. I have something to indicate in this article: "Allow the conversion between member function pointers, and The function is not allowed to be called after the conversion is completed, "is included in this rule, and this rule is incorporated into C standard. First, many popular compilers do not support this conversion (so conversion is standard requirements) However, it is not portable). Second, all compilers, if the conversion is successful, when calling the converted member function pointer, you can still achieve your expected function: The compiler does not have a so-called "undefined behavior (undefined behavior "This type of error is necessary (invocation) is feasible, but this is not a standard!). Third, allow conversion without allowing calls to be completely useless, only conversion and calling are feasible, can be convenient Effectively achieve entrustment, so that this language benefits. In order to make you confident that this is controversial, consider only the following code in a file, this code is legal: MSVC

Will not work, this is being used

MFC

You can only use a single inheritance. ) There are similar vulnerabilities (but not too serious) in Boost :: function. It seems that if you want to do anything more interesting about the member function pointer, you must prepare for challenges with this language vulnerability. If you want to deny the member function of C , it is difficult to design a defective point of view. In this article, I have something to indicate: "Allow the member function pointer to convert (CAST) without allowing the function to be called after the conversion is complete", incorporating this rule, incorporating this rule into C The standard is coming. First, a lot of popular compilers do not support this conversion (so conversion is standard requirements, but not portable). Second, all the compilers, if the conversion is successful, you can still achieve your expected function when calling the converted member function pointer: The compiler does not have the necessary "undefined behavior". " (Invocation) is feasible, but this is not a standard!). Third, allowing conversion without allowing calls to be completely useless, only converts and calls are feasible, to facilitate entrustment, so that this language benefits. In order to make you confident that this controversial argument, consider only the following code in a file, this code is legal: MFC

Message map macro (such as

ON_COMMAND

When you assemble a message

Id

And member functions pointers (types such as:

CCMDTARGET :: *

Sequence of member function pointers). this is

MFC

Class must inherit

CCMDTARGET

One of the reasons why you can handle the message. However, a variety of different message processing functions have different parameter lists (such as

OnDraw

The type of the first parameter of the process function is

CDC *

), So the sequence must contain a variety of different types of member function pointers.

MFC

How do you do this?

MFC uses a terrible compiler vulnerability (HACK) that puts all possible members function pointers in a huge union (Union), avoiding a C type matching check that is often required. (Look at it (look at it

AfxImpl.h

with

cmdtarg.cpp

Name

MessageMapFunctions

of

union

You will find this horror fact. )because

MFC

There is such an important part of the code, so the fact is that all the compilers have opened a green light for this vulnerability. (However, we will see later, if some classes use multiple inheritance, this vulnerability does not work in the MSVC, which is only the reasons for the use of single inheritance when using the MFC.) In Boost :: Similar vulnerabilities (but not too serious) in function. It seems that if you want to do anything more interesting about the member function pointer, you must prepare for challenges with this language vulnerability. If you want to deny the member function of C , it is difficult to design a defective point of view. In this article, I have something to indicate: "Allow the member function pointer to convert (CAST) without allowing the function to be called after the conversion is complete", incorporating this rule, incorporating this rule into C The standard is coming. First, a lot of popular compilers do not support this conversion (so conversion is standard requirements, but not portable). Second, all the compilers, if the conversion is successful, you can still achieve your expected function when calling the converted member function pointer: The compiler does not have the necessary "undefined behavior". " (Invocation) is feasible, but this is not a standard!). Third, allowing conversion without allowing calls to be completely useless, only converts and calls are feasible, to facilitate entrustment, so that this language benefits. In order to make you confident that this controversial argument, consider only the following code in a file, this code is legal: MSVC

Will not work, this is being used

MFC

You can only use a single inheritance. ) There are similar vulnerabilities (but not too serious) in Boost :: function. It seems that if you want to do anything more interesting about the member function pointer, you must prepare for challenges with this language vulnerability. If you want to deny the member function of C , it is difficult to design a defective point of view. In this article, I have something to indicate: "Allow the member function pointer to convert (CAST) without allowing the function to be called after the conversion is complete", incorporating this rule, incorporating this rule into C The standard is coming. First, a lot of popular compilers do not support this conversion (so conversion is standard requirements, but not portable). Second, all the compilers, if the conversion is successful, you can still achieve your expected function when calling the converted member function pointer: The compiler does not have the necessary "undefined behavior". " (Invocation) is feasible, but this is not a standard!). Third, allowing conversion without allowing calls to be completely useless, only converts and calls are feasible, to facilitate entrustment, so that this language benefits. In order to let you believe this controversial argument, consider only the following code in one file, this code is legal:

Boost :: function

There are similar vulnerabilities (but not too serious). It seems that if you want to do anything more interesting about the member function pointer, you must prepare for challenges with this language vulnerability. If you want to negate C

Member function pointer design has a defective point of view, it seems difficult. In this article, I have something to indicate: "Allow the member function pointer to convert (CAST) without allowing the function to be called after the conversion is complete", incorporating this rule, incorporating this rule into C The standard is coming. First, a lot of popular compilers do not support this conversion (so conversion is standard requirements, but not portable). Second, all the compilers, if the conversion is successful, you can still achieve your expected function when calling the converted member function pointer: The compiler does not have the necessary "undefined behavior". " (Invocation) is feasible, but this is not a standard!). Third, allowing conversion without allowing calls to be completely useless, only converts and calls are feasible, to facilitate entrustment, so that this language benefits. In order to let you believe this controversial argument, consider only the following code in one file, this code is legal:

CAST

), Not allowing the function to be called after the conversion is completed, "is included in this rule into

C

It is ridiculous. First, a lot of popular compilers do not support this conversion (so conversion is standard requirements, but not portable). Secondly, all compilers, if the conversion is successful, you can still achieve your expected function when calling the converted member function pointer: The compiler does not have a so-called "

undefined behavior

(Unexpected behavior) "The necessary necessary (calls (calls

Invocation

) Is feasible, but this is not a standard! ). Third, allowing conversion without allowing calls to be completely useless, only converts and calls are feasible, to facilitate entrustment, so that this language benefits. In order to let you believe this controversial argument, consider only the following code in one file, this code is legal:

Class SomeClass; TypeDef void (someclass :: * pclass, someclassFunction funcptr) {(PCLASS -> * Funcptr) ();

Note that the compiler must generate assembly code to call member function pointers, and the compiler does not know about the SomeClass class. Obviously, unless the linker performs some extreme fine optimization measures, the code will ignore the actual definition of the class and can operate correctly. And the direct consequences of this is that you can "safely" call the member function pointer converted from a completely different class. In order to explain the other half of my assertion - the conversion cannot be carried out in the way, I need to discuss how the compiler implements the member function pointer on the details. I will explain why the rules of the member function pointer have such a strict limit. Detailed discussion of member function pointers is not too easy, and everyone has learned a series of mistakes, so I carefully check the assembly code generated by a series of compilers ... (to be continued)

Someclass

There is no known class. Obviously, unless the linker performs some extreme fine optimization measures, the code will ignore the actual definition of the class and can operate correctly. And the direct consequences of this is that you can "safely" call the member function pointer converted from a completely different class. In order to explain the other half of my assertion - the conversion cannot be carried out in the way, I need to discuss how the compiler implements the member function pointer on the details. I will explain why the rules of the member function pointer have such a strict limit. Detailed discussion of member function pointers is not too easy, and everyone has learned a series of mistakes, so I carefully check the assembly code generated by a series of compilers ... (to be continued)

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

New Post(0)