第10章指针引言C程序设计中使用指针的好处使程序简洁、紧凑、高效有效地表示复杂的数据结构动态分配内存得到多于一个的函数返回值本章主要内容:地址和指针的概念对指针变量的操作和运算指针变量作为函数参数和函数返回值数组与指针指向函数的指针§10.1指针变量的概念与使用10.1.1变量与地址程序中:inti;floatk;内存中每个字节有一个编号-----地址…...…...2000200120022005内存02003ik编译或函数调用时为其分配内存单元变量是对程序中数据存储空间的抽象10.1.2地址与指针变量指针变量的值:其他变量的地址…...…...2000200420062005整型变量i10变量i_pointer2001200220032000变量i的地址指针i_pointer的值i的值10.1.3指针变量的定义一般形式:[存储类型]基类型*指针变量名;合法标识符指针变量本身的存储类型指针的目标变量的数据类型表示定义指针变量不是‘*’运算符例int*p1,*p2;float*q;staticchar*name;注意:1、int*p1,*p2;与int*p1,p2;2、指针变量名是p1,p2,不是*p1,*p23、指针变量只能指向定义时所规定类型的变量4、指针变量定义后,变量值不确定,应用前必须先赋值10.1.4指针变量的赋值取地址运算符&&变量名//表示变量的地址•单目运算符,优先级:2结合性:自右向左例如:inta;int*p=&a;取指针所代表地址的内容运算符*•单目运算符,优先级:2结合性:自右向左例如:*p=1;指针变量的初始化一般形式:[存储类型]数据类型*指针名=初始地址值;例inti;int*p=&i;变量必须已说明过类型应一致例int*p=&i;inti;例inti;int*p=&i;int*q=p;用已初始化指针变量作初值例main(){inti;staticint*p=&i;..............}()不能用auto变量的地址去初始化static型指针例main(){inti=10;int*p;*p=i;printf(“%d”,*p);}危险!例main(){inti=10,k;int*p;p=&k;*p=i;printf(“%d”,*p);}指针变量必须先赋值,再使用…...…...2000200420062005整型变量i10指针变量p200120022003随机§10.1.5对指针变量的操作直接访问:按变量地址存取变量值间接访问:通过存放变量地址的变量去访问变量例i=3;-----直接访问指针变量…...…...2000200420062005inti10int*i_pointer=&i20012002200320003例*i_pointer=20;-----间接访问20指针变量…...…...2000200420062005整型变量i10变量i_pointer2001200220032000整型变量k例1:inti,k,*i_pointer=&i;i=k;--直接访问k=*i_pointer;--间接访问10§指针变量与其所指向的变量之间的一些等价关系例:inti,*i_pointer=&i;3i2000i_pointer*i_pointeri*i_pointer&ii_pointeri=3;*i_pointer=33i2000i_pointer*i_pointeri*i_pointer&ii_pointeri=3;*i_pointer=3例2:inti=100,x,*p,*q;x=i;p=&i;x=*p;x=*(&i);q=&x;*q=i;*q=*p;指针变量定义时指定的基类型,应该与指针变量使用时指向的变量类型相同。main(){floatx=2.0,y;int*p;p=&x;y=*p;printf(%f,y);}修改为float*p;练习1请选出正确的程序段()。A)int*p;B)int*s,k;scanf(“%d”,p);*s=100;……C)int*s,k;D)int*s,k;char*p,c;char*p,e;s=&k;s=&k;p=&c;p=&e;*p='a';s=p;…*s=1;例3:通过指针变量访问整型变量Main(){inta,b;int*pointer_1,*pointer_2;a=100;b=10;pointer_1=&a;pointer_2=&b;printf(“%d,%d”,a,b);printf(“%d,%d”,*pointer_1,*pointer_2);}运行结果:100,10100,10定义两个指针变量代表两个指针变量分别指向的两个变量零指针与空类型指针零指针:(空指针)定义:指针变量值为零表示:int*p=0;p指向地址为0的单元,系统保证该单元不作它用表示指针变量值没有意义#defineNULL0int*p=NULL:p=NULL与未对p赋值不同用途:避免指针变量的非法引用在程序中常作为状态比较例int*p;......while(p!=NULL){...…}例4输入两个数,并使其从大到小输出main(){int*p1,*p2,*p,a,b;scanf(%d,%d,&a,&b);p1=&a;p2=&b;if(ab){p=p1;p1=p2;p2=p;}printf(a=%d,b=%d\n,a,b);printf(max=%d,min=%d\n,*p1,*p2);}运行结果:a=5,b=9max=9,min=5…...…...指针变量p1指针变量p20002008200220042006指针变量p2整型变量b整型变量a5200692008200620082006§10.1.7指针变量作为函数参数—地址传递特点:共享内存,“双向”传递swap(intx,inty){inttemp;temp=x;x=y;y=temp;}main(){inta,b;scanf(%d,%d,&a,&b);if(ab)swap(a,b);printf(\n%d,%d\n,a,b);}例5将数从大到小输出…...…...5变量a变量b(main)95运行结果:5,9值传递59变量temp变量y变量x(swap)95swap(int*p1,int*p2){intp;p=*p1;*p1=*p2;*p2=p;}main(){inta,b;scanf(%d,%d,&a,&b);if(ab)swap(&a,&b);printf(\n%d,%d\n,a,b);}…...20002002...59整型变量a整型变量b(main)(swap)指针p1指针p2整型p595例6将数从大到小输出地址传递运行结果:9,5地址传递也是值传递,只是值是地址20002002swap(int*p1,int*p2){int*p;*p=*p1;*p1=*p2;*p2=*p;}main(){inta,b;int*pointer_1,*pointer_2;scanf(%d,%d,&a,&b);pointer_1=&a;pointer_2=&b;if(ab)swap(pointer_1,pointer_2);printf(\n%d,%d\n,a,b);}编译警告!结果不对!intx;int*p=&x;x;例7将数从大到小输出…...20002008200A200220042006200C200E2010...59整型变量a整型变量b(main)指针pointer_1指针pointer_22000200220002002COPY(swap)指针p1指针p2指针p****假设2000指针变量在使用前必须赋值!main(){int*p1,*p2,*p,a,b;scanf(%d,%d,&a,&b);p1=&a;p2=&b;if(ab){p=p1;p1=p2;p2=p;}printf(a=%d,b=%d\n,a,b);printf(max=%d,min=%d\n,*p1,*p2);}voidmain(){voidswap(int*p1,int*p2);int*p1,*p2,a,b;scanf(%d,%d,&a,&b);p1=&a;p2=&b;if(ab)swap(p1,p2);printf(max=%d,min=%d\n,*p1,*p2);}voidswap(int*p1,int*p2){int*temp;temp=p1;p1=p2;p2=temp;}比较§10.1.6指针的算术运算:指针算术运算的结果依赖于指针的基类型pi指向地址pid(i为整型数,d为p指向的变量所占字节数)p++,p--,p+i,p-i,p+=i,p-=i等若p1与p2指向同一数组,p1-p2=两指针间元素个数(p1-p2)/dp1+p2无意义例p指向float数,则p+1p+14例p指向int型数组,且p=&a[0];则p+1指向a[1]例inta[10];int*p=&a[2];p++;*p=1;例inta[10];int*p1=&a[2];int*p2=&a[5];则:p2-p1=3;a[0]a[1]a[2]a[3]a[4]a[5]a[6]a[7]a[8]a[9]a数组pp+1,a+1p+i,a+ip+9,a+9指针变量的关系运算若p1和p2指向同一数组,则p1p2表示p1指的元素在前p1p2表示p1指的元素在后p1==p2表示p1与p2指向同一元素若p1与p2不指向同一数组,比较无意义p==NULL或p!=NULL§10.2指针与数组指向数组元素的指针变量例intarray[10];int*p;p=&array[0];//p=array;或int*p=&array[0];或int*p=array;array[0]array[1]array[2]array[3]array[9]...整型指针p&array[0]p数组名是表示数组首地址的地址常量数组元素表示方法a[0]a[1]a[2]a[3]a[9]...aa+9a+1a+2地址元素下标法a[0]a[1]a[2]a[9]a[0]a[1]a[2]a[3]a[9]...pp+9p+1p+2地址元素指针法*p*(p+1)*(p+2)*(p+9)[]变址运算符a[i]*(a+i)a[i]p[i]*(p+i)*(a+i)*a*(a+1)*(a+2)*(a+9)p[0]p[1]p[2]p[9]a[0]a[1]a[2]a[3]a[4]例8数组元素的引用方法main(){inta[5],*pa,i;for(i=0;i5;i++)a[i]=i+1;pa=a;for(i=0;i5;i++)printf(*(pa+%d):%d\n,i,*(pa+i));for(i=0;i5;i++)printf(*(a+%d):%d\n,i,*(a+i));for(i=0;i5;i++)printf(pa[%d]:%d\n,i,pa[i]);for(i=0;i5;i++)printf(a[%d]:%d\n,i,a[i]);}12345pa1.inta[]={1,2,3,4,5,6,7,8,9,10},*p=a,i;数组元素地址的正确表示:(A)&(a+1)(B)a++(C)&p(D)&p[i]数组名是地址常量p++,p--()a++,a--()a+1,*(a+2)()2.若有定义:int*p;则下列各选项中,不同于*p=*p+1;的是()A*p++;B.++*p;C.*p+=1;D.++(*p);练习2例voidmain(){inta[]={5,8,7,6,2,7,3};inty,*p=&a[1];y=(*--p)++;printf(“%d”,y);printf(“%d”,a[0]);}输出:56pp58762730123456a例9注意指针变量的运算6main(){inti,*p,a[7];p=a;for(i=0;i7;i++)scanf(%d,p++);printf(\n);fo