一天一个C++小知识
1. struct、enum、union三个关键字在C中定义比较麻烦,所以一般和typedef一块出现,而C++中则不用
2. struct和union中可以定义函数,但是:
2.1 struct和class用法完全相同,class有的功能它都用,唯一的区别在于当没有指定成员的访问权限时,struct中默认为public而class默认为private
2.2 union也支持三种访问权限的设置,但是:
2.2.1 union不支持继承
2.2.2 union不能定义虚函数
2.2.3 union和struct一样,没有指定访问权限时默认为public
2.2.4 union内的变量共享内存,所以不能用静态、引用类型
2.2.5 union中不是所有类都可以作为其成员变量,如果一个类(包括其父类)中还有自定义的constructor、copyconstructor、destructor、virtual functional或copy assignment operator(拷贝赋值运算符)任意一个,则不能
2.2.6 union匿名联合:定义union的时候不定以名字,仅仅通知编译器它的成员变量公用一个地址,并且变量本身直接引用,不使用“.”或者“->”,匿名联合体的用处,能想到是类中定义节省空间,此外可能实现自动类型转换,比如“union{void *p;int n;};int val;p=&val;”那么此时直接输出n就是一个地址值,而不用再把p显示转为int类型。匿名联合体有几个特点:
2.2.6.1 不能定义static
2.2.6.2 不能定义函数
2.2.6.3 不支持protected和private
2.2.6.4 全局域定义的匿名union只能是static否则必须放在匿名名字空间中。
3. struct 结构体
3.1 初始化:指定初始化A a={val1=2,val2=1;};;构造函数初始化。
3.2 直接访问:struct A{int val;int c[10];};A com;A.val=2;;间接访问:struct A *p;p->a;(也可以(*p).a但是不简洁)
3.3 自引用struct B{int val;struct B b;};不合法,循环下去永远不知道结构体的大小;struct B{int val;struct B *b;};合法,因为一个指针所占字节数是已知的。
3.4 跑题:int val;cout<<&val;可以,但是char cval;cout<<&cval;不行,因为&cval是一个char*变量,指向一个字符串,字符串需要以终止符\0结尾,但&cval没有,所以输出乱码。正确的方法是:1.printf(“%p”,&a);2.cout<<(void *)&cval;
4.union即所有成员变量共用一个内存首地址,联合体的赋值涉及一个大小端的问题:
4.1 大端模式:子数据的高字节存储在低地址,低字节存储在高地址
4.2 union型数据所占据的空间等于最大的成员所占的空间,对union型成员的存取都是相对于该联合体基地址偏移量为0开始的,这样通过union判断当前系统模式。union A{char ch[4];int I;} a;a.ch[0]=0x40;a.ch[1]=0x03;a.ch[2]=0x02;a.ch[3]=0x11;printf(“%x”,test.i);数组中下标小的地址也低,内存内容依次为04、03、02、11一共四个字节,而不分类型直接打印十六进制,应该从内存高地址到低地址看,0x11020304,低位04在低地址上。
5.结构体的内存对齐。
5.1 struct A{char ch;int ival;double dval;}a;len1=sizeof(a)执行这一句代码,len1值为16而不是1+4+8,这是因为结构体中元素是按照定义顺序一个一个放到内存中的,但并不是紧密排列。从结构体的首地址开始,每一个元素放置到内存中,它都会认为内存是以它自己的大小来划分的,因此,元素放置的位置一定会在自己宽度的整数倍上开始。如果用||表示分隔,则十六个长度应该为|ch(char)占一个|NULL|NULL|NULL|ival(int)占四个|dval(double)占八个|。
5.2 strcut B{char ch;double dval;int ival;}b;len2=sizeof(b)执行这一代码,按照上述规则,len2理应为8+8+4=20,但实际输出24。这是因为,在经过上面的规则分析之后,检查计算出的存储单元是否为所有元素中最宽的元素的长度的整数倍,是的话结束,否则补齐它的整数倍。
5.3 这样,我们在定义结构体时候,内部的元素应该按所占字节数从大到小排列,并且考虑补位的影响。
6. enum
6.1 默认以int类型存储,占四个字节,可以存储的最大值0xffffff。但是我们可以通过继承的方式改变枚举的大小。C++中规定0<=sizeof(enum)<=sizeof(int),编译器究竟是分配多少内存,emmmmm~
6.2 另外,enum Test{a=1,b,c=5,d,e}test;test=e;cout<<test;输出结果为7,因为enum会默认给后面未幅值的再最近一个幅值的基础上加1。
想入门C语言?看这里
C语言编程基础
http://www.makeru.com.cn/live/1758_311.html?s=143793
C语言 数组和字符串
http://www.makeru.com.cn/video/2238_12037.html?s=143793
C语言 指针专题一
http://www.makeru.com.cn/video/2239_12043.html?s=143793
C语言可控制led灯
http://www.makeru.com.cn/live/1392_304.html?s=143793