c++對象切片

什麼事對象切片:

  c++在將一個派生類轉換爲基類的過程中,派生類的一部分將被基類接收不到,只能留下基類大小的對象。


傳值調用的切片:


#include<iostream>
using namespace std;
class Base
{
public:
virtual void func1()
{
cout << "Base::func1" << endl;
}
virtual void func2()
{
cout << "Base::func2" << endl;
}
private:
int a;
};
class Derive :public Base
{
public:
virtual void func1()
{
cout << "Derive::func1" << endl;
}
virtual void func3()
{
cout << "Derive::func3" << endl;
}
virtual void func4()
{
cout << "Derive::func4" << endl;
}
private:
int b;
};
typedef void(*FUNC) ();
void PrintVTable(int* VTable)
{
cout << " 虛表地址>" << VTable << endl;
for (int i = 0; VTable[i] != 0; ++i)
{
printf(" 第%d個虛函數地址 :0X%x,->", i, VTable[i]);
FUNC f = (FUNC)VTable[i];
f();
}
cout << endl;
}
void function(Base b)
{
b.func1();
}
void Test1()
{
Base b1;
Derive d1;
int* VTable1 = (int*)(*(int*)&b1);
int* VTable2 = (int*)(*(int*)&d1);
PrintVTable(VTable1);
PrintVTable(VTable2);
function(d1);
}
int main()
{
Test1();
system("pause");
}

wKioL1dEBsPw8WpkAAAjKs-tdsk046.png


wKioL1dECJPS0QYsAAAo58KSjzc275.png

  再拷貝的過程中發生了切片,在調用構造函數的時候初始化VPTR指向基類的VTABLE,並且只拷貝了對象的基類部分,所以最後就變成了一個基類的對象。

  如果要防止這種現象的發生,只需把基類定義成純虛函數就可以了。




將派生類的指針傳遞給基類

#include<iostream>
using namespace std;
class Base
{
public:
virtual void func1()
{
cout << "Base::func1" << endl;
}
virtual void func2()
{
cout << "Base::func2" << endl;
}
private:
int a;
};
class Derive :public Base
{
public:
virtual void func1()
{
cout << "Derive::func1" << endl;
}
virtual void func3()
{
cout << "Derive::func3" << endl;
}
virtual void func4()
{
cout << "Derive::func4" << endl;
}
private:
int b;
};
typedef void(*FUNC) ();
void PrintVTable(int* VTable)
{
cout << " 虛表地址>" << VTable << endl;
for (int i = 0; VTable[i] != 0; ++i)
{
printf(" 第%d個虛函數地址 :0X%x,->", i, VTable[i]);
FUNC f = (FUNC)VTable[i];
f();
}
cout << endl;
}
void function(Base* b)
{
b->func1();
b->func2();
}
void Test1()
{
Base b1;
Derive d1;
int* VTable1 = (int*)(*(int*)&b1);
int* VTable2 = (int*)(*(int*)&d1);
PrintVTable(VTable1);
PrintVTable(VTable2);
function((Base*)&d1);
}
int main()
{
Test1();
system("pause");
}


wKioL1dEC0qDlPbcAAAiKsn7IZk444.png


wKiom1dECsDB09p4AAAtOtuJ8j4810.png

解釋:

   在將派生類的指針傳給基類之後,就將派生類的大小看作是積累的大小,所以就只能訪問基類的大小,由於沒有發生構造函數,所以VPTR還是派生類的。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章