C/C++杂记:深入理解数据成员指针、函数成员指针
1. 数据成员指针
对于普通指针变量来说,其值是它所指向的地址,0表示空指针。
而对于数据成员指针变量来说,其值是数据成员所在地址相对于对象起始地址的偏移值,空指针用-1表示。例:
代码示例:
struct X { int a; int b; }; #define VALUE_OF_PTR(p) (*(long*)&p) int main() { int X::*p = 0; // VALUE_OF_PTR(p) == -1 p = &X::a; // VALUE_OF_PTR(p) == 0 p = &X::b; // VALUE_OF_PTR(p) == 4 return 0; }
2. 函数成员指针
函数成员指针与普通函数指针相比,其size为普通函数指针的两倍(x64下为16字节),分为:ptr和adj两部分。
(1) 非虚函数成员指针
ptr部分内容为函数指针(指向一个全局函数,该函数的第一个参数为this指针),adj部分始终为0。例:
代码示例:
extern "C" int printf(const char*, ...); struct B { void foo() { printf("B::foo(): this = 0x%p ", this); } }; struct D : public B { void bar() { printf("D::bar(): this = 0x%p ", this); } }; void (B::*pbfoo)() = &B::foo; // ptr: points to _ZN1B3fooEv, adj: 0 void (D::*pdfoo)() = &D::foo; // ptr: points to _ZN1B3fooEv, adj: 0 void (D::*pdbar)() = &D::bar; // ptr: points to _ZN1D3barEv, adj: 0 extern "C" void _ZN1B3fooEv(B*); extern "C" void _ZN1D3barEv(D*); #define PART1_OF_PTR(p) (((long*)&p)[0]) #define PART2_OF_PTR(p) (((long*)&p)[1]) int main() { printf("&B::foo->ptr: 0x%lX ", PART1_OF_PTR(pbfoo)); printf("&B::foo->adj: 0x%lX ", PART2_OF_PTR(pbfoo)); // 0 printf("&D::foo->ptr: 0x%lX ", PART1_OF_PTR(pdfoo)); printf("&D::foo->adj: 0x%lX ", PART2_OF_PTR(pdfoo)); // 0 printf("&D::bar->ptr: 0x%lX ", PART1_OF_PTR(pdbar)); printf("&D::bar->adj: 0x%lX ", PART2_OF_PTR(pdbar)); // 0 D* d = new D(); d->foo(); _ZN1B3fooEv(d); // equal to d->foo() d->bar(); _ZN1D3barEv(d); // equal to d->bar() return 0; }