C ++ pointer directly calls class member functions

Source: Internet
Author: User

In programming, we often encounter the requirement to call member functions through function pointers in a "class". For example, when a class is usedC ++When the sorting function qsort in the standard library requires a "comparison function" pointer for the qsort parameter, if this "class" uses a member function as a "comparison function ", you need to pass the pointer of this member function to qsort for its call. The member functions used to call a class with pointers in this article include the following three situations:

1) Assign the member function pointer of the class to the non-member function pointer of the same type, for example:

Example 1

 
 
  1. # Include <stdlib. h>
  2. Typedef void (* Function1) (); // defines a function pointer type.
  3. Function1 f1;
  4. Class Test1
  5. {
  6. Public:
  7. //... The called member function.
  8. Void Memberfun1 () {printf ("% s \ n", "Calling Test3: Memberfun2 OK ");};//
  9. Void Memberfun2 ()
  10. {
  11. F1 = reinterpret_cast <Function1> (Memberfun1); // assign the member function pointer to f1. Compilation error.
  12. F1 ();
  13. }
  14. //...
  15. };
  16. Int main ()
  17. {
  18. Test1 t1;
  19. T1.Memberfun2 ();
  20. Return 0;
  21. }

2) within a "class", there are standard library functions, such as qsort, or other global functions that use function pointers to call member functions of the class. For example:

Example 2:

 
 
  1. # Include <stdlib. h>
  2. Class Test2
  3. {
  4. Private:
  5. Int data [2];
  6. //...
  7. Public:
  8. //...
  9. Int _ cdecl Compare (const void * elem1, const void * elem2) // member function.
  10. {
  11. Printf ("% s \ n", "Calling Test2: Memberfun OK ");
  12. Return * (int *) elem1)-* (int *) elem2 );
  13. }
  14. Void Memberfun ()
  15. {
  16. Data [0] = 2; data [1] = 5;
  17. Qsort (data, 2, sizeof (int), Compare); // The standard library function calls
  18. // Member function. Compilation error.
  19. }
  20. //...
  21. };
  22. Int main ()
  23. {
  24. Test2 t2;
  25. T2.Memberfun (); // call a member function.
  26. Return 0;
  27. }

3) within the same "class", one member function calls another member function, for example:

Example 3:

 
 
  1. # Include "stdlib. h"
  2. Class Test3
  3. {
  4. Public:
  5. //...
  6. Void Memberfun1 (void (* f2) () {f2 () ;}// member function 1 calls member function // 2.
  7. Void Memberfun2 () {printf ("% s \ n", "Calling Test3: Memberfun2 OK") ;}// member function 2.
  8. Void Memberfun3 () {Memberfun1 (Memberfun2);} // compilation Error
  9. //...
  10. };
  11. Int main ()
  12. {
  13. Test3 t3;
  14. T3.Memberfun3 (); // call a member function.
  15. Return 0;
  16. }

There are no significant errors in the code syntax in the above three cases. In some earlier compilation environments, such as VC ++ 4.0, the code can usually be compiled, or at most give a question reminder (Warning ). Later compilation tools, such as VC ++ 6.0 and some other commonly used C ++ compilation software, cannot be compiled through the above Code, the error is shown as follows (in the third case, VC ++ 6.0 is used as an example ):

Error C2664: 'memberfun1': cannot convert parameter 1 from 'void (void) 'to 'void (_ cdecl *) (void )'
None of the functions with this name in scope match the target type

That is, the function type called in the Memberfun1 parameter is incorrect.

As prompted above, errors cannot be eliminated only by changing the function type. However, if you extract these functions from the class definition, you can eliminate errors by compiling without making any changes, take the third case as an example. The following code can be compiled:

 
 
  1. # Include <stdlib. h>
  2. Void Memberfun1 (void (* f2) () {f2 () ;}// the original member function 1 calls the member function // 2.
  3. Void Memberfun2 () {printf ("% s \ n", "Calling Test3: Memberfun2 OK") ;}// original member function 2.
  4. Void Memberfun3 () {Memberfun1 (Memberfun2 );}
  5. Int main ()
  6. {
  7. Memberfun3 ();
  8. Return 0;
  9. }

1st and 2 are the same as 3rd.

It can be concluded that the above three situations cannot be compiled because the function type call is not correct, but related to the class. If the compilation fails, the function pointer is used to call the member functions of the "class". By compiling, the function pointer is used to call non-member functions, and the function types are identical. So, what are the differences between the member function pointer and the non-member function pointer of the "class?

In the following program, the sizeof () function can be used to view the size of member function pointers and non-member function pointers of various "classes" and output them to the screen.

 
 
  1. # Include "stdafx. h"
  2. # Include <iostream>
  3. # Include <typeinfo. h>
  4. Class Test; // an undefined class.
  5. Class Test2 // an empty class.
  6. {
  7. };
  8. Class Test3 // a defined class.
  9. {
  10. Public:
  11. //...
  12. Void (* memberfun )();
  13. Void Memberfun1 (void (* f2) () {f2 () ;}// member function 1 calls member function // 2.
  14. Void Memberfun2 (); // member function 2.
  15. //...
  16. };
  17. Class Test4: virtual Test3, Test2 // a derivative class with virtual inheritance ).
  18. {
  19. Public:
  20. Void Memberfun1 (void (* f2) () {f2 ();}
  21. };
  22. Class Test5: Test3, Test2 // an inherited class derivative class ).
  23. {
  24. Public:
  25. Void Memberfun1 (void (* f2) () {f2 ();}
  26. };
  27.  
  28. Int main ()
  29. {
  30. Std: cout <"General function pointer length =" <sizeof (void (*) () <'\ n ';
  31. Std: cout <"-member function pointer length of the Class-" <'\ n ';
  32. Std: cout <"Test3 member function pointer length =" <sizeof (void (Test3 ::*)()) <'\ n ';
  33. Std: cout <"Test5 class member function pointer length =" <sizeof (void (Test5: *) () <'\ n ';
  34. Std: cout <"Test4 member function pointer length =" <sizeof (void (Test4: *) () <'\ n ';
  35. Std: cout <"Test class member function pointer length =" <sizeof (void (Test: *) () <'\ n ';
  36. Return 0;
  37. }

The output result is compiled by VC ++ 6.0 and runs on the Win98 operating system. Other operating systems may be different ):

  • Generally, the length of a non-member function pointer is 4.
  • -Class member function pointer length-
  • Test3 member function pointer length = 4
  • Test5 member function pointer length = 8
  • Test4 member function pointer length = 12
  • Test member function pointer length = 16

The above results show that in 32-bit Win98 operating system, the length of the function pointer is generally 4 bytes and 32 bits ), the length of the member function pointer of the class changes with the definition of the class or the inheritance type and relationship of the class, from the 4-byte 32-bit of the class Test3 without inheritance relationship) to the 12-byte 96-bit of the Virtual Inheritance class Test4), only the class Test that is not defined in declaration) because some information related to it is unclear that the member function pointer can contain a maximum of 16 bytes and 128 bits ). Obviously, unlike the general function pointer, the pointer to the member function of the "class" contains not only the information of the member function address, but also information related to the class attributes. Therefore, generally, the function pointer and the member function pointer of the class are two different types. Of course, you cannot directly call the member function of the class using the general function pointer, this is why compilation fails in the three cases mentioned in this article. Although earlier versions of the compilation software can still be compiled, this poses a serious risk to the program.

As for why the pointer to the member function of the class is also different in length, from 32-bit to 128-bit, the difference is great, since I did not see Official Microsoft documents, I can only speculate that VC ++ 6.0 optimized the member function pointer of the class during compilation to shorten the pointer length as much as possible, after all, using a 128-bit or 96-bit pointer on a 32-bit operating system will affect program performance. However, no matter how optimized, the class member function pointer contains a certain amount of object Objects) information is determined. If other operating systems and compilation software are processed Similarly, you can use the above programs to verify themselves.

So how do I use pointers to call member functions of a class when necessary? You can consider the following methods:

1) set the member function to be called to the static type. For example, in Example 2, add the following static symbol before the Compare definition of the class Test2 member function to change it ):

 
 
  1. Class Test2
  2. {
  3. //....
  4. Int static _ cdecl Compare (const void * elem1, const void * elem2) // member function.
  5. // Others remain unchanged
  6. }

The changed code is successfully compiled. The reason is that member functions of the static type are separated from classes, and their function pointers do not contain object information, which is consistent with general function pointers. This method is simple, but has two disadvantages: 1. The called function member definition cannot contain any class members, including variables and functions. 2. Because static members are used, the class is limited when it is inherited.

2) use a static member function whose function parameters contain object information to call other member functions indirectly. Example 3, modify the class Test3 as follows to modify the Italic), and the main function remains unchanged, so that compilation can be successful:

 
 
  1. Class Test3
  2. {
  3. Public:
  4. //...
  5. Void static _ cdecl Helper (Test3 * test3)
  6. {
  7. Test3-> Memberfun2 ();
  8. }
  9. Void Memberfun1 (void (* f2) (Test3 *) {f2 (this);} // transmits the object information to the Helper function.
  10. Void Memberfun2 () {printf ("% s \ n", "Calling Test3: Memberfun2 OK") ;}// member function 2.
  11. Void Memberfun3 () {Memberfun1 (Helper );}
  12. //...
  13. };

This indirect method has no limitations on member functions and overcomes the disadvantages that member functions cannot use any class members in the first method. However, due to static members, class inheritance is still restricted.

(3) Use a global function) As a member function of the intermediate indirect call class. For example, use example 3 to compile the Code as follows ):

 
 
  1. Class Test3;
  2. Void _ cdecl Helper (Test3 * test3 );
  3. Class Test3
  4. {
  5. Public:
  6. //...
  7. Void Memberfun1 (void (* f2) (Test3 *) {f2 (this);} // member function 1 calls member function // 2.
  8. Void Memberfun2 () {printf ("% s \ n", "Calling Test3: Memberfun2 OK") ;}// member function 2.
  9. Void Memberfun3 () {Memberfun1 (Helper );}
  10. //...
  11. };
  12.  
  13. Void _ cdecl Helper (Test3 * test3)
  14. {
  15. Test3-> Memberfun2 ();
  16. };

This method has no requirements for member functions, but requires more code.

In addition to the above three methods, there are other methods, such as modifying the code at the Assembly level to solve the above problems, which is not within the scope of this article.

Conclusion: function pointers cannot directly call member functions of a class. Indirect methods are required because member function pointers are fundamentally different from common function pointers, in addition to the address information, the member function pointer carries the information of the object to which it belongs. This document provides three methods for indirectly calling member functions. These three methods have their own advantages and disadvantages and are suitable for different occasions.

We hope that the above content will help you.

Contact Us

The content source of this page is from Internet, which doesn't represent Alibaba Cloud's opinion; products and services mentioned on that page don't have any relationship with Alibaba Cloud. If the content of the page makes you feel confusing, please write us an email, we will handle the problem within 5 days after receiving your email.

If you find any instances of plagiarism from the community, please send an email to: info-contact@alibabacloud.com and provide relevant evidence. A staff member will contact you within 5 working days.

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.