目录
1.指针的内存布局
先看下面的例子:
int *p;
大家都知道这里定义了一个指针p,但是p到底是什么东西呢?它到底占多大的内存空间呢?
我们说任何一种数据类型都可以把它当作一个模子,p毫无疑问是某个模子“咔”出来的,经过sizeof测试(32位系统):sizeof(p)的值为4.那就说明“咔”出p的这个模子大小为4字节。显然,这个模子不是“int”,虽然它大小也是4.既然不是“int”,那就一定是“int *”了,接下来我们这么理解这个定义:
一个“int *”类型的模子在内存上“咔”出了4字节的空间,然后把这个4字节大小的空间命名为p,同时限定这4字节的空间里面只能存储某个内存地址,即使你存入别的任何数据,都将被当作地址处理,而且这个内存地址开始的连续的四个字节上只能存储某个int类型的数据。总体来说,指针是什么呢?指针就是地址。
如上图所示,我们把p称为指针变量,p里存储的内存地址处的内存称为p所指向的内存。指针变量p里存储的任何数据都被当作地址来处理。
2.“*”与防盗门的钥匙
这里这个“*”号怎么理解呢?举个例子:当你回到家门口,你想进屋第一件事情就是拿出钥匙来开锁,试想防盗门的锁芯是不是很像这个“*”号?你要进屋必须要用钥匙,那你去读/写一块内存是不是也要一把钥匙?这个“*”号是不是就是我们最好的要是?使用指针的时候,没有它,你是不可能读/写某块内存的。
3.int *p=NULL和*p=NULL的区别
先看下面的代码:
int *p = NULL;
这时候我们可以通过调试器查看p的值为0x00000000。这句代码的意思是:定义一个指针变量p,其指向的内存里面保存的是int类型的数据;在定义变量p的同时把p的值设置成0x00000000,而不是把*p的值设置成0x00000000。这个过程叫做初始化,是在编译的时候进行的。
明白了什么是初始化之后,在看下面的代码:
int *p ;
*p = NULL;
同样,我们可以在调试器上调试这两行代码。
第一行代码,定义了一个指针变量p,其指向的内存里面保存的是int类型的数据;但是这时候变量p本身的值是多少不得而知,也就是说现在变量p保存的有可能是一个非法的地址。
第二行代码,给*p赋值为NULL,即给p指向的内存赋值为NULL;但是由于p指向的内存可能是非法的,所以调试的时候调试器可能会报告一个内存访问错误。这样的话,我们可以把上面的代码进行改写,使p指向一块合法的内存:
int i = 10;
int *p = &i;
*p = NULL;
在调试器上调试一下就会发现p指向的内存由原来的10变为0了,而p本身的值,即内存地址并没有改变。
4.如何将数值存储到指定的内存地址
假设现在需要往内存地址0x12ff7c上存入一个整型数0x100,那么怎么才能做到呢?
我们知道可以通过一个指针向其指向的内存地址写入数据,那么这里的内存地址0x12ff7c其本质不就是一个指针嘛,所以我们使用下面的方法:、
int *p = (int *)0x12ff7c;
*p = 0x100;
需要注意的是,将地址0x12ff7c赋值给指针变量p的时候必须强制转换。
将上面两行代码转为一行表示:
*(int*)0x12ff7c=0x100;