4.1.7 POD对象
在C++中,我们把传统的C风格的struct叫做POD(Plain Old Data)对象。一般来说,POD对象应该满足如下特性。
对于POD类型T的对象,不管这个对象是否拥有类型T的有效值,如果将该对象的底层字节序列复制到一个字符数组(或者无符号字符数组)中,再将其复制回对象,那么该对象的值与原始值一样。
对于任意的POD类型T,如果两个T指针分别指向两个不同的对象obj1和obj2,如果用memcpy库函数把obj1的值复制到obj2,那么obj2将拥有与obj1相同的值。
简言之,针对POD对象,其二进制内容是可以随便复制的,在任何地方,只要其二进制内容在,就能还原出正确无误的POD对象。对于任何POD对象,都可以使用memset()函数或者其他类似的内存初始化函数。
现在动手
为了更好地理解POD对象的含义,我们体验一下如何采用memxxx()函数对POD对象进行存储与还原。
编写Win32控制台程序,主程序如下:
【程序 4-3】使用memxxx函数操作POD对象
- 01 #include "stdafx.h"
- 02 #include <cstring>
- 03
- 04
- 05 struct PERSON
- 06 {
- 07 char _name[16];
- 08 int _age;
- 09 bool _gender;
- 10 };
- 11
- 12 void print(PERSON * p)
- 13 {
- 14 printf("%s,%d,%s\r\n", p->_name, p->_age, (p->_gender ? "男" : "女"));
- 15 }
- 16
- 17 int main()
- 18 {
- 19
- 20 PERSON p1 = {"佟湘玉", 28, false};
- 21 PERSON p3 = {"白展堂", 26, true};
- 22 print(&p1);
- 23 print(&p3);
- 24
- 25
- 26 char bytes[sizeof(PERSON)];
- 27 memcpy(bytes, &p1, sizeof(PERSON));
- 28
- 29 PERSON p2;
- 30 memset(&p2, 0, sizeof(PERSON));
- 31 print(&p2);
- 32
- 33
- 34 memcpy(&p2, bytes, sizeof(PERSON));
- 35 print(&p2);
- 36
- 37
- 38 memcpy(&p2, &p3, sizeof(PERSON));
- 39 print(&p2);
- 40
- 41 return 0;42 }
结果输出如图4-5所示。
|
(点击查看大图)图4-5 运行结果 |
因此,对于POD对象,我们完全可以大胆地使用memxxx函数进行操作,从而完成对对象复制、赋值的目的。但是注意,对于多态类的对象,要慎重考虑使用memset,因为它会同时修改vtable指针!vtable指针是多态的根本所在,弄乱了对象的虚表指针,很有可能会酿成大错。