一、C语言的字符串类型
1、C语言没有原生字符串类型
- 很多高级语言像java、C#等就有字符串类型,有个String来表示字符串,用法和int这些很像,可以String s1 = "linux";来定义字符串类型的变量。
- C语言没有String类型,C语言中的字符串是通过字符指针来间接实现的。
2、C语言使用指针来管理字符串
- C语言中定义字符串方法:char *p = "linux";
- 此时p就叫做字符串,但是实际上p只是一个字符指针(本质上就是一个指针变量,只是p指向了一个字符串的起始地址而已)。
3、C语言中字符串的本质
(1)指针指向头、固定尾部的地址相连的一段内存;
(2)字符串就是一串字符。
- 字符反映在现实中就是文字、符号、数字等人用来表达的字符,反映在编程中字符就是字符类型的变量。
- C语言中使用ASCII编码对字符进行编程,编码后可以用char型变量来表示一个字符。字符串就是多个字符打包在一起共同组成的。
- 字符串在内存中其实就是多个字节连续分布构成的(类似于数组,字符串和字符数组非常像)。
- C语言中字符串有3个核心要点:第一是用一个指针指向字符串头;第二是固定尾部(字符串总是以'\0'来结尾);第三是组成字符串的各字符彼此地址相连。
- 字符'\0'是一个ASCII字符,其实就是ASCII编码为0的那个字符(数字0有它自己的ASCII编码48)。要注意区分'\0'和'0'和0。('\0'等于0,'0'等于48)
- '\0'作为一个特殊的数字被字符串定义为(幸运的选为)结尾标志。产生的副作用就是:字符串中无法包含'\0'这个字符。(C语言中不可能存在一个包含'\0'字符的字符串),这种思路就叫“魔数”(魔数就是选出来的一个特殊的数字,这个数字表示一个特殊的含义,你的正式内容中不能包含这个魔数作为内容)。
4、指向字符串的指针和字符串本身
- char *p = "linux";
- p本质上是一个字符指针,占4字节;
- "linux"分配在代码段,占6个字节;
5、存储多个字符的2种方式
- 字符串和字符数组
6、字符数组初始化与sizeof、strlen
(1)sizeof是C语言的一个关键字,也是C语言的一个运算符(sizeof使用时是sizeof(类型或变量名)
- sizeof运算符用来返回一个类型或者是变量所占用的内存字节数。
- 为什么需要sizeof?主要原因一是int、double等原生类型占几个字节和平台有关;二是C语言中除了ADT之外还有UDT,这些用户自定义类型占几个字节无法一眼看出,所以用sizeof运算符来让编译器帮忙计算。
- sizeof(数组名)得到的永远是数组的元素个数(也就是数组的大小),和数组中有无初始化,初始化多、少等是没有关系的;
(2)strlen是一个C语言库函数
- 这个库函数的原型是:size_t strlen(const char *s);这个函数接收一个字符串的指针,返回这个字符串的长度(以字节为单位)。
- strlen返回的字符串长度是不包含字符串结尾的'\0'的。
- 因为从字符串的定义(指针指向头、固定结尾、中间依次相连)可以看出无法直接得到字符串的长度,需要用strlen函数来计算得到字符串的长度。
- strlen是用来计算字符串的长度的,只有传递合法的字符串进去才有意义,如果随便传递一个字符指针,但是这个字符指针并不是字符串是没有意义的。
7、字符串初始化与sizeof、strlen
8、字符数组与字符串的本质差异(内存分配角度)
(1)字符数组char a[] = "linux"
- 定义了一个数组a,数组a占6字节,右值"linux"本身只存在于编译器中,编译器将它用来初始化字符数组a后丢弃掉(也就是说内存中是没有"linux"这个字符串的);
- 这句就相当于是:char a[] = {'l', 'i', 'n', 'u', 'x', '\0'};
(2)字符串char *p = "linux"
- 定义了一个字符指针p,p占4字节,分配在栈上;同时还定义了一个字符串"linux",分配在代码段;然后把代码段中的字符串(一共占6字节)的首地址(也就是'l'的地址)赋值给p。
(3)字符数组和字符串有本质差别
- 字符数组本身是数组,数组自身自带内存空间,可以用来存东西(所以数组类似于容器);而字符串本身是指针,本身永远只占4字节,而且这4个字节还不能用来存有效数据,所以只能把有效数据存到别的地方,然后把地址存在p中。也就是说字符数组自己存那些字符;字符串一定需要额外的内存来存那些字符,字符串本身只存真正的那些字符所在的内存空间的首地址。
二、C语言之结构体概述
1、结构体类型是一种自定义类型
- C语言中的2种类型:原生类型和自定义类型。
2、从数组到结构体的进步之处
- 数组有2个明显的缺陷:第一个是定义时必须明确给出大小,且这个大小在以后不能再更改;第二个是数组要求所有的元素的类型必须一致。更复杂的数据结构中就致力于解决数组的这两个缺陷。
- 结构体是用来解决数组的第二个缺陷的,可以将结构体理解为一个其中元素类型可以不相同的数组。结构体完全可以取代数组,只是在数组可用的范围内数组比结构体更简单。
3、如何访问结构体变量中的元素?
(1)数组中元素的访问方式:表面上有2种方式(数组下标方式和指针方式);实质上都是指针方式访问。
(2)C语言规定用结构体变量来访问元素用. ;用结构体变量的指针来访问元素用->。
(3)其实本质上还是用指针来访问的。
4、什么是结构体对齐访问?
- 结构体中元素的访问其实本质上还是用指针方式,结合这个元素在整个结构体中的偏移量和这个元素的类型来进行访问的。
- 但是因为结构体要考虑元素的对齐访问,所以每个元素实际占的字节数和自己本身的类型所占的字节数不一定完全一样。
- 一般来说,我们用.的方式来访问结构体元素时,我们是不用考虑结构体的元素对齐的。因为编译器会帮我们处理这个细节。
- 但是因为C语言本身是很底层的语言,而且做嵌入式开发经常需要从内存角度,以指针方式来处理结构体及其中的元素,因此还是需要掌握结构体对齐规则。
5、结构体为何要对齐访问
- 结构体中元素对齐访问主要原因是为了配合硬件,也就是说硬件本身有物理上的限制,如果对齐排布和访问会提高效率,否则会大大降低效率。
- 内存本身是一个物理器件(DDR内存芯片,SoC上的DDR控制器),本身有一定的局限性:如果内存每次访问时按照4字节对齐访问,那么效率是最高的。
- Cache的一些缓存特性,还有其他硬件(譬如MMU、LCD显示器)的一些内存依赖特性,要求内存对齐访问。
- 对齐访问牺牲了内存空间,换取了速度性能;而非对齐访问牺牲了访问速度性能,换取了内存空间的完全利用。
6、结构体对齐的规则和运算
(1)编译器可以设置内存对齐的规则
- 32位编译器,一般编译器默认对齐方式是4字节对齐。
(2)gcc支持但不推荐的对齐指令:#pragma pack() #pragma pack(n) (n=1/2/4/8)
- #pragma用来设置编译器的对齐方式。
- #pragma pack()设置编译器1字节对齐;#pragma pack(n),括号中的数字表示多少字节对齐。
- 需要#prgama pack(n)开头,以#pragma pack()结尾,定义一个区间,这个区间内的对齐参数就是n。
- #prgma pack的方式在很多C环境下都是支持的,但是gcc不建议使用。
(3)gcc推荐的对齐指令__attribute__((packed)) __attribute__((aligned(n)))
- __attribute__((packed))放在要进行内存对齐的类型定义的后面,起作用的范围只有加了此修饰的这一个类型。packed的作用就是取消对齐访问。
- __attribute__((aligned(n)))放在要进行内存对齐的类型定义的后面,起作用的范围只有加了此修饰的这一个类型。它的作用是让整个结构体变量整体进行n字节对齐(注意是结构体变量整体n字节对齐,而不是结构体内各元素也要n字节对齐)
(4)规则
- 见博客http://blog.csdn.net/oqqhutu12345678/article/details/65437355
三、offsetof宏与container_of宏
1、offsetof宏
(1)offsetof宏的作用:用宏来计算结构体中某个元素和结构体首地址的偏移量(其实质是通过编译器来帮我们计算)。
(2)offsetof宏的原理:虚拟一个type类型结构体变量,然后用type.member的方式来访问那个member元素,继而得到member相对于整个变量首地址的偏移量。
(3)学习思路:第一步先学会用offsetof宏,第二步再去理解这个宏的实现原理。
- (TYPE *)0 :这是一个强制类型转换,把0地址强制类型转换成一个指针,这个指针指向一个TYPE类型的结构体变量。(实际上这个结构体变量可能不存在,但是只要我不去解引用这个指针就不会出错)。
- ((TYPE *)0)->MEMBER (TYPE *)0是一个TYPE类型结构体变量的指针,通过指针指针来访问这个结构体变量的member元素。
- &((TYPE *)0)->MEMBER 等效于&(((TYPE *)0)->MEMBER),意义就是得到member元素的地址。但是因为整个结构体变量的首地址是0,
2、container_of宏
- 见博客http://blog.csdn.net/oqqhutu12345678/article/details/72851192
四、共用体union
1、共用体类型的定义、变量定义和使用
(1)共用体union和结构体struct在类型定义、变量定义、使用方法上很相似。
(2)共用体和结构体的不同
- 结构体类似于一个包裹,结构体中的成员彼此是独立存在的,分布在内存的不同单元中,只是被打包成一个整体叫做结构体而已;
- 共用体中的各个成员其实是一体的,彼此不独立,他们使用同一个内存单元。可以理解为:有时候是这个元素,有时候是那个元素。更准确的说法是同一个内存空间有多种解释方式。
- 共用体union就是对同一块内存中存储的二进制的不同的理解方式。
- 在有些书中把union翻译成联合(联合体),这个名字不好。现在翻译成共用体比较合适。
(3)union的sizeof测到的大小实际是union中各个元素里面占用内存最大的那个元素的大小。
(4)union中的元素不存在内存对齐的问题
- 因为union中实际只有1个内存空间,都是从同一个地址开始的(开始地址就是整个union占有的内存空间的首地址),所以不涉及内存对齐。
2、共用体和结构体的相同和不同
- 相同点就是操作语法几乎相同。
- 不同点是本质上的不同。struct是多个独立元素(内存空间)打包在一起;union是一个元素(内存空间)的多种不同解析方式。
3、共用体的主要用途
- 共用体就用在那种对同一个内存单元进行多种不同规则解析的这种情况下。
- C语言中其实是可以没有共用体的,用指针和强制类型转换可以替代共用体完成同样的功能,但是共用体的方式更简单、更便捷、更好理解。
五、枚举
1、枚举的作用
- 枚举在C语言中其实是一些符号常量集。直白点说:枚举定义了一些符号,这些符号的本质就是int类型的常量,每个符号和一个常量绑定。这个符号就表示一个自定义的一个识别码,编译器对枚举的认知就是符号常量所绑定的那个int类型的数字。
- 枚举中的枚举值都是常量,怎么验证?
- 枚举符号常量和其对应的常量数字相对来说,数字不重要,符号才重要。符号对应的数字只要彼此不相同即可,没有别的要求。所以一般情况下我们都不明确指定这个符号所对应的数字,而让编译器自动分配。(编译器自动分配的原则是:从0开始依次增加。如果用户自己定义了一个值,则从那个值开始往后依次增加)
2、C语言为何需要枚举
- C语言没有枚举是可以的。使用枚举其实就是对1、0这些数字进行符号化编码,这样的好处就是编程时可以不用看数字而直接看符号。符号的意义是显然的,一眼可以看出。而数字所代表的含义除非看文档或者注释。
- 而宏定义的目的是不用数字而用符号。从这里可以看出:宏定义和枚举有内在联系。宏定义和枚举经常用来解决类似的问题,他们俩基本相当可以互换,但是有一些细微差别。
3、宏定义和枚举的区别
- 枚举是将多个有关联的符号封装在一个枚举中,而宏定义是完全散的。也就是说枚举其实是多选一。
- 什么情况下用枚举?当我们要定义的常量是一个有限集合时(譬如一星期有7天,譬如一个月有31天,譬如一年有12个月····),最适合用枚举。(其实宏定义也行,但是枚举更好)
- 不能用枚举的情况下(定义的常量符号之间无关联,或者无限的)用宏定义。
- 宏定义先出现,用来解决符号常量的问题;后来发现定义的符号常量彼此之间有关联(多选一的关系),用宏定义来做虽然可以但是不贴切,于是发明枚举来解决这种情况。
4、枚举的定义和使用
六、大小端模式
1、什么是大小端模式
(1)大端模式(big endian)和小端模式(little endian)。
(2)在串口等串行通信中
- 一次只能发送1个字节。这时候要发送一个int类型的数就遇到一个问题。int类型有4个字节,我是按照byte0 byte1 byte2 byte3这样的顺序发送,还是按照byte3 byte2 byte1 byte0这样的顺序发送。规则就是发送方和接收方必须按照同样的字节顺序来通信,否则就会出现错误。这就叫通信系统中的大小端模式。这是大小端这个词和计算机挂钩的最早问题。
(3)更多是指计算机存储系统的大小端。
- 在计算机内存/硬盘/Nnad中。因为存储系统是32位的,但是数据仍然是按照字节为单位的。于是乎一个32位的二进制在内存中存储时有2种分布方式:高字节对应高地址(大端模式)、高字节对应低地址(小端模式)
(4)大端模式和小端模式本身没有优劣
- 理论上按照大端或小端都可以,但是要求必须存储时和读取时按照同样的大小端模式来进行,否则会出错。
(5)有些CPU公司用大端(譬如C51单片机);有些CPU用小端(譬如ARM)
- 当不知道当前环境是用大端模式还是小端模式时就需要用代码来检测当前系统的大小端。
- 经典笔试题:用C语言写一个函数来测试当前机器的大小端模式。
2、用union来测试机器的大小端模式
3、指针方式来测试机器的大小端
4、看似可行实则不行的测试大小端方式:位与、移位、强制类型转化
- 位与运算是编译器提供的运算,这个运算是高于内存层次的(或者说&运算在二进制层次具有可移植性,也就是说&的时候一定是高字节&高字节,低字节&低字节,和二进制存储无关)。
- 因为C语言对运算符的级别是高于二进制层次的。右移运算永远是将低字节移除,而和二进制存储时这个低字节在高位还是低位无关的。
5、通信系统中的大小端(数组的大小端)
(1)譬如要通过串口发送一个0x12345678给接收方,但是因为串口本身限制,只能以字节为单位来发送,所以需要发4次;接收方分4次接收,内容分别是:0x12、0x34、0x56、0x78.接收方接收到这4个字节之后需要去重组得到0x12345678(而不是得到0x78563412)。
(2)所以在通信双方需要有一个默契,就是:先发/先接的是高位还是低位?这就是通信中的大小端问题。
(3)在通信协议中,大小端是非常重要的,大家使用别人定义的通信协议还是自己要去定义通信协议,一定都要注意标明通信协议中大小端的问题。
七、内存来源
1、栈(stack)、堆(heap)、数据区(.data)
- 在一个C语言程序中,能够获取的内存就是三种情况:栈(stack)、堆(heap)、数据区(.data)
2、栈的详解
- 运行时自动分配&自动回收:栈是自动管理的,程序员不需要手工干预。方便简单。
- 反复使用:栈内存在程序中其实就是那一块空间,程序反复使用这一块空间。
- 脏内存:栈内存由于反复使用,每次使用后程序不会去清理,因此分配到时保留原来的值。
- 临时性:(函数不能返回栈变量的指针,因为这个空间是临时的)。
- 栈会溢出:因为操作系统事先给定了栈的大小,如果在函数中无穷尽的分配栈内存总能用完。
3、堆内存详解
- 操作系统堆管理器管理:堆管理器是操作系统的一个模块,堆管理内存分配灵活,按需分配。
- 大块内存:堆内存管理者总量很大的操作系统内存块,各进程可以按需申请使用,使用完释放。
- 程序手动申请&释放:手工意思是需要写代码去申请malloc和释放free。
- 脏内存:堆内存也是反复使用的,而且使用者用完释放前不会清除,因此也是脏的。
- 临时性:堆内存只在malloc和free之间属于我这个进程,而可以访问。在malloc之前和free之后都不能再访问,否则会有不可预料的后果。
(1)堆内存使用范例
- void *是个指针类型,malloc返回的是一个void *类型的指针,实质上malloc返回的是堆管理器分配给我本次申请的那段内存空间的首地址(malloc返回的值其实是一个数字,这个数字表示一个内存地址)。为什么要使用void *作为类型?主要原因是malloc帮我们分配内存时只是分配了内存空间,至于这段空间将来用来存储什么类型的元素malloc是不关心的,由我们程序自己来决定。
- 什么是void类型。早期被翻译成空型,这个翻译非常不好,会误导人。void类型不表示没有类型,而表示万能类型。void的意思就是说这个数据的类型当前是不确定的,在需要的时候可以再去指定它的具体类型。void *类型是一个指针类型,这个指针本身占4个字节,但是指针指向的类型是不确定的,换句话说这个指针在需要的时候可以被强制转化成其他任何一种确定类型的指针,也就是说这个指针可以指向任何类型的元素。
- malloc的返回值:成功申请空间后返回这个内存空间的指针,申请失败时返回NULL。所以malloc获取的内存指针使用前一定要先检验是否为NULL。
- malloc申请的内存时用完后要free释放。free(p);会告诉堆管理器这段内存我用完了你可以回收了。堆管理器回收了这段内存后这段内存当前进程就不应该再使用了。因为释放后堆管理器就可能把这段内存再次分配给别的进程,所以你就不能再使用了。
- 再调用free归还这段内存之前,指向这段内存的指针p一定不能丢(也就是不能给p另外赋值)。因为p一旦丢失这段malloc来的内存就永远的丢失了(内存泄漏),直到当前程序结束时操作系统才会回收这段内存。
(2)malloc的一些细节表现
- malloc(0):malloc申请0字节内存本身就是一件无厘头事情,一般不会碰到这个需要。如果真的malloc(0)返回的是NULL还是一个有效指针?答案是:实际分配了16Byte的一段内存并且返回了这段内存的地址。这个答案不是确定的,因为C语言并没有明确规定malloc(0)时的表现,由各malloc函数库的实现者来定义。
- malloc(4):gcc中的malloc默认最小是以16B为分配单位的。如果malloc小于16B的大小时都会返回一个16字节的大小的内存。malloc实现时没有实现任意自己的分配而是允许一些大小的块内存的分配。
- malloc(20)去访问第25、第250、第2500····会怎么样。实战中:120字节处正确,1200字节处正确····终于继续往后访问总有一个数字处开始段错误了。
4、代码段、数据段、bss段
(1)编译器在编译程序的时候,将程序中的所有的元素分成了一些组成部分,各部分构成一个段,所以说段是可执行程序的组成部分。
(2)代码段
- 代码段就是程序中的可执行部分,直观理解代码段就是函数堆叠组成的。
(3)数据段(也被称为数据区、静态数据区、静态区)
- 数据段就是程序中的数据,直观理解就是C语言程序中的全局变量。(注意:全局变量才算是程序的数据,局部变量不算程序的数据,只能算是函数的数据)。
(4)bss段(又叫ZI(zero initial)段)
- bss段的特点就是被初始化为0,bss段本质上也是属于数据段,bss段就是被初始化为0的数据段。
- 数据段(.data)和bss段的区别和联系:二者本来没有本质区别,都是用来存放C程序中的全局变量的。区别在于把显示初始化为非零的全局变量存在.data段中,而把显式初始化为0或者并未显式初始化(C语言规定未显式初始化的全局变量值默认为0)的全局变量存在bss段。
(5)有些特殊数据会被放到代码段
- C语言中使用char *p = "linux";定义字符串时,字符串"linux"实际被分配在代码段,也就是说这个"linux"字符串实际上是一个常量字符串而不是变量字符串。
- const型常量:C语言中const关键字用来定义常量,常量就是不能被改变的量。const的实现方法至少有2种:第一种就是编译将const修饰的变量放在代码段去以实现不能修改(普遍见于各种单片机的编译器);第二种就是由编译器来检查以确保const型的常量不会被修改,实际上const型的常量还是和普通变量一样放在数据段的(gcc中就是这样实现的)。
(5)显式初始化为非零的全局变量和静态局部变量放在数据段
- 放在.data段的变量有2种:第一种是显式初始化为非零的全局变量。第二种是静态局部变量,也就是static修饰的局部变量。(普通局部变量分配在栈上,静态局部变量分配在.data段)
(6)未初始化或显式初始化为0的全局变量放在bss段
- bss段和.data段并没有本质区别,几乎可以不用明确去区分这两种。
5、总结
(1)相同点
- 三种获取内存的方法,都可以给程序提供可用内存,都可以用来定义变量给程序用。
(2)不同点
- 栈内存对应C中的普通局部变量(别的变量还用不了栈,而且栈是自动的,由编译器和运行时环境共同来提供服务的,程序员无法手工控制);
- 堆内存完全是独立于我们的程序存在和管理的,程序需要内存时可以去手工申请malloc,使用完成后必须尽快free释放。(堆内存对程序就好象公共图书馆对于人);
- 数据段对于程序来说对应C程序中的全局变量和静态局部变量。
(3)不同的存储方式有不同的特点,简单总结如下:
- 函数内部临时使用,出了函数不会用到,就定义局部变量
- 堆内存和数据段几乎拥有完全相同的属性,大部分时候是可以完全替换的。但是生命周期不一堆内存的生命周期是从malloc开始到free结束,而全局变量是从整个程序一开始执行就开始,直到整个程序结束才会消灭,伴随程序运行的一生。启示:如果你这个变量只是在程序的一个阶段有用,用完就不用了,就适合用堆内存;如果这个变量本身和程序是一生相伴的,那就适合用全局变量。