C语言:指向指针的指针

来源:互联网 发布:西安莲湖区行知小学 编辑:程序博客网 时间:2024/04/29 06:07
 

 一. 指针概念:
程序申明变量如下:
short int i;
char a;
short int * pi;
程序会在内存某地址空间上为各变量开辟空间,如下图所示。
内存地址→6     7      8     9     10     11    12    13     14    15
-------------------------------------------------------------------------------------
…  |     |      |      |      |      |       |      |      |      |    
-------------------------------------------------------------------------------------
     |short int i |char a|      |short int * pi|
图中所示中可看出:
i 变量在内存地址5的位置,占两个字节。
a变量在内存地址7的位置,占一个字节。
pi变量在内存地址9的位置,占两个字节。(注:pi 是指针,我这里指针的宽度只有两个字节,32位系统是四个字节)
接下来如下赋值:
i=50;
pi=&i;
经过上在两句的赋值,变量的内存映象如下:
内存地址→6     7      8     9     10     11    12    13      14     15
--------------------------------------------------------------------------------------
…  |    50      |      |      |    6         |      |      |       |    
--------------------------------------------------------------------------------------
     |short int i |char a|      |short int * pi|
可以看到:短整型指针变量pi的值为6,它就是I变量的内存起始地址。所以,这时当我们对*pi进行读写操作时,其实就是对i变量的读写操作。如:
*pi=5;   //就是等价于i=5; 
 
  
二. 指针的地址与指向另一指针地址的指针
指针变量本身与其它变量一样也是在某个内存地址中的,如pi的内存起始地址是10。同样的,我们也可能让某个指针指向这个地址。
看下面代码:
short int * * ppi;    //这是一个指向指针的指针,注意有两个*号
ppi=π
  
第一句:short int * * ppi;——申明了一个指针变量ppi,这个ppi是用来存储(或称指向)一个short int * 类型指针变量的地址。
第二句:&pi那就是取pi的地址,ppi=π就是把pi的地址赋给了ppi。即将地址值10赋值给ppi。如下图:
内存地址→6     7      8     9     10     11    12    13       14    15
------------------------------------------------------------------------------------
…  |    50     |      |      |      6       |       10      |      |    
------------------------------------------------------------------------------------
     |short int i|char a|      |short int * pi|short int ** ppi|
从图中看出,指针变量ppi的内容就是指针变量pi的起始地址。于是……
ppi的值是多少呢?——10。
*ppi的值是多少呢?——6,即pi的值。
**ppi的值是多少呢?——50,即I的值,也是*pi的值。
呵呵!

  
三. 应用实例
1. 设计一个函数:void find(char table[ ], char find, char * ps)
要求:这个函数参数中的数组table是以0值为结束的字符串,要求在字符串table中查找字符是参数find里的字符。如果找到,函数通过第三个参数(ps)返回值为table字符串中第一个找到的字符的地址。如果没找到,则为ps为0。
实现代码如下: 
void find(char table[ ], char find, char ** ps)
{
    int i;
    for (i=0;*(table+i)!=0;i++)
    {
       if (*(table+i)==find)
       {
         *ps=table+i
         break;
       }
       else if (*(table+i)==0)
       {
         *ps=0;
         break;
       }
    }
}

主函数:

void main()
{
   char num[]={"lijuanjuanhappy\0"}; 
   char a=’p’; 

   char * p=0; 

   find(str,a,p);  
   if (0==p )
   {
      printf (“没找到!\n”);//1.如果没找到则输出此句
   }
   else
   {
      printf(“找到了,p=%d”,p);  //如果找到则输出此句
   }
}

在分析之前注意:

       在C语言中,实参向对形参的数据传递是“值传递”,单向传递,只由实参传给形参,而不能由形参传回来给实参。在内存中,实参单元与形参单元是不同的单元。在调用函数时,给形参分配存储单元,并将实参对应的值传递给形参,调用结束后,形参单元被释放,实参单元仍保留并维持原值。

分析:
调用函数时的整个操作如下: 
   table=str; 
    find=a;
    ps=&p;    //以上三句是调用时隐含的动作。
    int i;
    for (i=0;*(table+i)!=0;i++)
    {
       if (*(table+i)==find)
       {
         *ps=table+i
         break;
       }
       else if (*(table+i)==0)
       {
         *ps=0;
         break;
       }
    }
看明白了吗?
ps指向指针p的地址。
对*ps的修改就是对p值的修改。