C++多重继承与void*指针转换问题

C++支持多重继承,然而多重继承可能会导致一些奇怪的问题,我前段时间遇到一个指针转换问题,非常典型。

先看一个简单的测试代码:

#include <iostream>
using namespace std;

class IA
{
public:
    virtual ~IA(){}
    virtual void a() = 0;
};

class IB
{
public:
    virtual ~IB(){}
    virtual void b() = 0;
};

class CMulti : public IA, public IB
{
public:
    CMulti(){}
    ~CMulti(){}

    void a(){ cout << "C::a()" << endl; }
    void b(){ cout << "C::b()" << endl; }
};

void testCastA(void *p)
{
    cout << "cast from void*(" << p << ")to IA*:  ";
    IA *a = (IA *)p;
    a->a();
}

void testCastB(void *p)
{
    cout << "cast from void*(" << p << ")to IB*:  ";
    IB *b = (IB *)p;
    b->b();
}

int _tmain(int argc, _TCHAR* argv[])
{
    CMulti * c = new CMulti;

    cout << "cast to void*, then to IA or IB:" << endl;
    testCastA((void*)c);
    testCastB((void*)c);
    cout << endl;

    cout << "static_cast to void*, then to IA or IB:" << endl;
    testCastA((void*)static_cast<IA*>(c));
    testCastB((void*)static_cast<IB*>(c));
    cout << endl;

    cout << "dynamic_cast to void*, then to IA or IB:" << endl;
    testCastA((void*)dynamic_cast<IA*>(c));
    testCastB((void*)dynamic_cast<IB*>(c));

    return 0;
}

我测试了:

  • 强制转换为void*
  • 先static_cast再强制转换为void*
  • 先dynamic_cast再强制转换为void*

(32位程序)运行的结果如下图所示:

差异很明显了,结论也很明了:多重继承时,子类指针转换为非第一继承的父类指针时,会发生地址偏移(注意图上标红的部分)。这是因为每一个父类都会占用 4 个字节维护自己的虚函数表。所以,当 CMulti* 转换为 IB* 时,指针加 4 ,因为 IA 是 CMulti 的第一父类, IB 是第二父类,依次类推……

如果我们在某些地方不得不使用 void* 来进行代码适配时,遇到多重继承就要注意这一点,否则很可能你调用的是 b() 方法,实际执行的是 a() ,达不到预期效果。

时间: 2024-11-08 17:46:20

C++多重继承与void*指针转换问题的相关文章

C++多重继承下一个子类和父类指针转换的bug

这两天有个C++新手问了我一个问题,他的工程当中有一段代码执行不正确,不知道是什么原因.我调了一下,代码如果精简下来,大概是下面这个样子: class IBaseA { public: virtual void fnA() = 0; int m_nTestA; }; class IBaseB { public: virtual void fnB() = 0; int m_nTestB; }; class CTest : public IBaseA,public IBaseB { public:

void指针的转换(2)

void类型指针能够通过显式转换为具有更小或同样存储对齐限制的指针.但数据可能失真.所谓"同样存储对齐限制"是指void类型指针所指的数据在内存中所占的长度与显式转换后的指针所指的数据在内存中所占的长度相等,比方以上程序中的p1所指的原数据在内存中占2个字节,p2所指的数据在内存中也是占两个数据. 但应注意的是.仅仅有上面的这样的转换前后指针所指数据类型一致的转换才保持数据不失真,假设类型不一致,即使具有同样存储对齐限制,也有可能失真,比方由short转向unsigned short,

子类指针转换成父类指针时,指针值会变化

指针转换在C++中再正常不过了,特别是子类向父类的转换,一般情况下转换后指针值是不会变的.但是某些特殊情况,子类指针转换成父类指针,指针值会产生一定的偏移量,至于为什么就不清楚了.要确保指针值不变,可以先将指针转成void*,再根据实际情况强制转换成需要的类型就可以了. 本文地址:http://www.cnblogs.com/vcpp123/p/5902866.html 参考文章: C++多重继承下的指针类型转换 http://www.veryhuo.com/a/view/52953.html

C++ Primer 学习笔记与思考_7 void和void*指针的使用方法

(一)void的含义 void的字面意思是"无类型",void差点儿仅仅有"凝视"和限制程序的作用,由于从来没有人会定义一个void变量,让我们试着来定义: void a; 这行语句编译时会出错.提示"illegal use of type 'void'".只是.即使void a的编译不会出错.它也没有不论什么实际意义. void真正发挥的作用在于: (1) 对函数返回的限定: (2) 对函数參数的限定. int f(void); equal t

void及void指针类型

1.概述 许多初学者对C/C++语言中的void及void指针类型不甚理解,因此在使用上出现了一些错误.本文将对void关键字的深刻含义进行解说,并 详述void及void指针类型的使用方法与技巧. 2.void的含义 void的字面意思是“无类型”,void *则为“无类型指针”,void *可以指向任何类型的数据. void几乎只有“注释”和限制程序的作用,因为从来没有人会定义一个void变量,让我们试着来定义: void a; 这行语句编译时会出错,提示“illegal use of ty

void和void*指针的一些理解

void 和 void* 指针分别表示无类型和无类型指针. void 的作用是限制: 1,函数无返回值. 2,函数无参数. 当函数的返还值无参数的时候一定要加上 void ,因为在缺省的状态下函数的返还值是 int ,并且如果不需要的时候就一定要将他声明成 void 类型这样看着更明显.当函数无参数的时候在声明的时候也要加上 void 的这也是一个好习惯,增强了可读性,当然在 K&R C 中是允许在声明的时候省去参数的. void 类型的和指针可以接受任何类型的类型的指针的赋值(但是不代表这个v

void类型及void指针

1.概述 许多初学者对C/C 语言中的void及void指针类型不甚理解,因此在使用上出现了一些错误.本文将对void关键字的深刻含义进行解说,并 详述void及void指针类型的使用方法与技巧. 2.void的含义 void的字面意思是“无类型”,void *则为“无类型指针”,void *可以指向任何类型的数据. void几乎只有“注释”和限制程序的作用,因为从来没有人会定义一个void变量,让我们试着来定义: void a; 这行语句编译时会出错,提示“illegal use of typ

void指针

1回顶部 热门文章:C++中extern “C”含义深层探索 编程实现盗2005版QQ源码 1.概述 许多初学者对C/C++语言中的void及void指针类型不甚理解,因此在使用上出现了一些错误.本文将对void关键字的深刻含义进行解说,并详述void及void指针类型的使用方法与技巧. 2.void的含义 void的字面意思是“无类型”,void *则为“无类型指针”,void *可以指向任何类型的数据. void几乎只有“注释”和限制程序的作用,因为从来没有人会定义一个void变量,让我们试

转载--void指针(void *的用法)

转自:jimmy 指针有两个属性:指向变量/对象的地址和长度 但是指针只存储地址,长度则取决于指针的类型 编译器根据指针的类型从指针指向的地址向后寻址 指针类型不同则寻址范围也不同,比如: int*从指定地址向后寻找4字节作为变量的存储单元 double*从指定地址向后寻找8字节作为变量的存储单元 1.void指针是一种特别的指针    void *vp    //说它特别是因为它没有类型    //或者说这个类型不能判断出指向对象的长度 2.任何指针都可以赋值给void指针    type *