OS_MEM *OSMemCreate (void *addr, INT32U nblks, INT32U blksize, INT8U *err);中有如下语句:
OS_MEM *pmem;
INT8U *pblk;
void **plink;
INT32U i;
plink = (void **)addr;
pblk = (INT8U *)addr + blksize;
for (i = 0; i < (nblks - 1); i++) {
*plink = (void *)pblk;
plink = *plink;
pblk = pblk + blksize;
其中,为什么在使用的时候,addr明明是一个一级指针,但是在对plink赋值的时候就把addr强制的转换成了二维指针?
查了很多技术书籍,网上看了很多文章,现在终于懂了。
二级指针就是指向指针的指针,他的内容是一个目标变量的地址,也就是说仍然是一个指针,对二级指针取两次内容才能取到目标变量的内容。
1.、分析:plink = (void **)addr;
在声明的时候,plink是二维指针,在这里将addr强制的转换成二维指针再赋值给plink的原因是:让addr以前指向的内容让编译器解释成地址,也就是一个指针,如果不做这个强制转换,以前addr指向的内容就不是一个地址,也就是不是指针,在这个函数当中,我们想把addr指向的一维数组,分割成大小相同的若干块,就必须用指针把它们链接起来,所以将addr强制转换成二级指针,然后赋值给plink,将它转换成二级指针的目的是这样的:首先,addr是一个指针,它指向申请内存的起始地址,在这里,我们想要addr行使两个角色,1是它的定义,就是指向申请内存的起始地址,2是在addr指向的变量内存放的数据也当做一个地址,让这个数据指向下一个内存块的起始地方,为了让addr指向的变量的数据当做地址,只能强行将该指针转换成二级指针。然后让plink去执行连接的操作,也就是在以前addr指向的地方放上指针;plink本身是存放在栈上的,plink这个符号的值是指向addr的,*plink就是取plink指向地址单元的内容,而plink指向地址单元的内容是一个地址,即是一个指针,plink指向地址单元的内容也就是addr指向地址单元的内容,但由于addr是一维指针,所以它指向的内容不会被解释成一个地址,而是一般的内容。现在*plink就是把addr所在存储单元的内容解释成一个指针,并且将下一个block的首地址赋值给此存储单元,理解了这点就可以理解下面的源代码了,同时对C里面指针的概念又有了进一步的认识。
2、*plink = (void *)pblk;
对二级指针plink执行取内容操作,注意此时plink是一个二级指针,前面加上*,表示成为一个一级指针,即*plink就是一个一级指针了(也就是*plink这个变量存入pblk的值,下一个block开始的地址),它指向的地方改为下一个blcok开始的地址处,也就是pblk的值。
3、plink = *plink;
Plink也指向下一个block开始的地方。
4、pblk = pblk + blksize;
Pblk不断的下移,以指向不同的block开始处。
总结:进行强制转换的目的其实就是为了把所指向的地方转换成一个指针。
void *pblk;
执行操作:pmem->OSMemFreeList =*(void **)pblk;意味着取出pblk的内容,由于pblk被强制转换成了二维指针,所以它的内容就不是一般的值,而是一个指针。
要将一个常数变成一个指针,int *p =(int *)0x01ffeeee;
要将一个常数变成一个二维指针,int **q =(int *)0x01ffeeee;此时*p就是一个一维指针了
http://blog.csdn.net/uestczhangchao/article/details/5589476