一、数据的存储
大小端介绍
大端字节序:把数据的低位字节序的内容放在高地址处,高位字节序的内容放在低地址处。
小端字节序:把数据的低位字节序的内容放在低地址处,高位字节序的内容放在高地址处。
设计一个小程序来判断当前机器的字节序。
二、指针
1.字符指针
字符指针是将一个常量字符串的首字符的地址存放到指针变量中。
str3和str4指向的是一个常量字符串,而常量字符串在内存中只会存一份,且不能被修改。
2.数组指针
复习指针数组:
int* arr1[10];//整型指针的数组
char* arr2[4];//一级字符指针的数组
char** arr3[5];//二级指针的数组
数组指针是指针。
int (*p)[10];
p先和*结合,说明p是一个指针变量,指向的数组有10个元素,每个元素的类型是int。
&arr 表示的是数组的地址,而不是数组首元素的地址,例如 int(*p)[10],是一个数组指针,数组的地址+1,跳过整个数组,所以&arr+1相对于&arr的差值是40。
int(*parr[10])[5];
//parr是一个存放数组指针的数组,该数组能够存放10个数组指针
//每个数组指针能够指向一个数组,数组有5个元素,每个元素的类型是int
3.函数指针
void(*p)();
p首先和*结合,说明p是一个指针,指向的是一个函数,( )内是指向的函数的参数,void是返回值类型。
注意:数组名 != &数组名
函数名 == &函数名
代码1:
(*(void(*)())0)();
1)void(*)( ):表示的是一个函数指针;
2)(void(*)( ))0:对0进行强制类型转换,转换成一个函数指针;
3)*(void(*)( ))0:对0地址进行了解引用操作;
4)(*(void(*)( ))0)( ):调用0地址处的函数。
代码2:
void (*signal(int , void(*)(int)))(int);
1)signal首选和( )结合,说明signal是一个函数,函数的第一个参数的类型是int,第二个参数的类型是一个函数指针;
2)signal函数的返回类型也是一个函数指针,该指针指向一个参数为int,返回类型为void的函数。
//对void(*)(int)的函数指针类型重命名为pfun_t
typedef void(*pfun_t)(int);
//代码2就可以写成
pfun_t signal(int, pfun_t);
4.函数指针数组
把函数的地址存放到一个数组中,这个数组就叫函数指针数组。
int (*parr[10])();
parr首先和( )结合,说明parr是一个数组,数组的内容是int (*)( )类型的函数指针。
函数指针数组的用途:转移表。
5.指向函数指针数组的指针
指向函数指针数组的指针是一个指针,指针指向一个数组。数组的元素都是函数指针。
//整型数组
int arr[5];
int (*p1)[5] = &arr;
//整型指针数组
int* arr[5];
int* (*p2)[5] = &arr;
//p2是指向整型指针数组的指针
//函数指针数组
//&函数指针数组
int(*p)(int,int);//函数指针
int(*p2[4])(int,int);//函数指针数组
int(*(p3[4])(int,int) = p2;//取出的是函数指针数组的地址
//p3是一个指向函数指针数组的指针