一、内存地址
1 什么是内存地址
address 唯一性·连续性·编号从小~大
字节是最小存储单位 每个字节都有自己的地址编号
指针==地址
2 地址相关运算:&取变量所占字节的首地址 * 根据地址取值
* &为互逆运算
#include <stdio.h>
int main()
{
int a = 5;
scanf_s("%d", &a);
printf("%d %d %d\n", a,&a,*&a);//得到a变量所在内存中所有字节的最小的地址编号 首地址
return 0;
}
二、指针变量与地址
1 什么是指针变量:存储内存地址的变量为指针变量
声明指针变量
int a = 5;
int* p;
p = &a;
int* 是一个类型(int指针/地址 类型 整形指针类型)
2 指针变量的字节数:根据操作系统的位数而不同。
x86 32位操作系统 最多支持4G内存实际为3.25G 指针变量字节数为4。
x64 64位操作系统 支持4G 8G 16G 32G 64G 128G 256G内存理论上可以无限支持 指针变量字节数为8。
printf("%d %d %d %d\n", sizeof(p), sizeof(*p), p, p + 1);
sizeof(p):指针变量p内存大小/所储的地址编号的大小
sizeof(*p):*p指向的空间/指向的空间大小
p:地址编号
p+1:地址编号加上一个int的字节数
3 指针的移动:根据数据类型不同,移动的步伐大小也不同
char* pc = &a;
double* pd = &a;
short* ps = &a;
printf("%d %d %d %d\n", sizeof(p), sizeof(*p), p, p + 1);
printf("%d %d %d %d\n", sizeof(pc), sizeof(*pc), pc, pc + 1);
printf("%d %d %d %d\n", sizeof(pd), sizeof(*pd), pd, pd + 1);
printf("%d %d %d %d\n", sizeof(ps), sizeof(*ps), ps, ps + 1);
指针变量+1:指向下一个地址,所加的1为指针类型的字节数
4 利用指针对存储值进行算术运算(注意运算符的优先级)
*p = 3;
printf("%d\n", a);
*p += 5;
printf("%d\n", a);
++*p;
printf("%d\n", a);
(*p)--;//因为*跟--是同一个优先级,要从右往左算
//*p--;
printf("%d\n", a);
因为*跟--是同一个优先级,要从右往左算,不能写成*p--;
三、指针注意事项
注意1: 指针不要位移到不属于本程序的内存空间
也不要利用指针改变不属于本程序的内存空间的数据
int a, b;
int* p = &a;
*p = 8;
p = &b;
*p = 5;
//p += 10;
//*p = 6;//非法行为
//p = (int*)0x12314234;
//*p = 6;//非法行为
注意2:声明多个指针变量类型时,*不能省略。
int a, b, c;
int* p, q, m;
int* p1, * q1, * m1;
注意3: 指针变量的初始值如果没有明确指向目标用NULL赋值
int age = 0;//
int* point = NULL;
无类型指针(不确定类型) 空地址(空指针) 没有地址
void 空/占位符号
五、多级指针
多级指针又称为:指向指针的指针。
#include <stdio.h>
int main05()
{
int a = 5;
int* p = &a;
int** pp = &p;
int*** ppp = &pp;
printf("%d", ***ppp);
return 0;
}