第8章指针8.1指针的基本概念8.2指针与一维数组8.3指针与二维数组8.4指针与字符串8.5指针数组8.6多级指针8.7指针与函数小结C程序设计中使用指针可以:使程序简洁、紧凑、高效有效地表示复杂的数据结构动态分配内存得到多于一个的函数返回值8.1指针的基本概念程序中:shorti;floatk;内存中每个字节有一个编号-----地址…...…...2000200120022005内存02003ik编译或函数调用时为其分配内存单元变量是对程序中数据存储空间的抽象8.1.1存储单元、内存地址及指针指针就是存储单元的地址。一个变量的指针就是该变量在内存中的存储单元的地址。存取变量中的数据直接引用:按变量名称存取变量值的方式。间接引用:将某变量a的指针(地址)放在另一个变量pa中,通过对变量pa的值(a变量的地址),由此地址存取变量a的方式。10910001:a10910001:a10001pa例如l8-1d1:#includestdio.hvoidmain(){intx,*px;floaty;x=10;y=23.5;px=&x;printf(%d%.1f,x,y);printf(%d%.1f,*px,y);}指针变量px……1023.5内容地址图8.2变量、地址关系示意图2000变量x变量y2000200120022003200420052006…...…...2000200420062005整型变量i10变量i_pointer2001200220038.1.2指针变量指针:一个变量的地址指针变量:专门存放变量地址的变量2000指针指针变量变量的内容变量的地址指针变量变量变量地址(指针)变量值指向地址存入指针变量变量的指针和指向变量的指针变量变量的指针:就是变量的在内存中的存储单元的地址。指针变量:存放(指向某个变量的)地址的变量。用运算符*表示指针变量所指向的对象。如:指针变量pa指向变量a。用*pa表示pa的对象,即变量a。(*pa与a为同一个变量)下例两语句作用相同a=345;*pa=345;10910001:a10001pa10001:a*pa10001pa345指针部分指针:就是变量的地址。指针变量:是可存放指针的变量。指针类型:由变量的类型决定。例:指针变量pa指向变量a,即表明变量pa中存放着变量a的地址。pa=&a;*pa与a等价10001:a*pa10001pa345一般形式:数据类型*指针名;合法标识符指针的目标变量的数据类型表示定义指针变量不是‘*’运算符例int*p1,*p2;float*q;staticchar*name;注意:1、int*p1,*p2;与int*p1,p2;不同,每一个指针变量前有*2、指针变量名是p1,p2,不是*p1,*p23、指针变量只能指向定义时所规定类型的变量int*p,a;charc;floatx;p只能指向a,不能指向c或x。4、指针变量定义后,变量值不确定即指向不确定,称为悬挂指针,不能间接引用悬挂指针。应用前必须先赋值5、指针变量只能存放地址,不能存放非地址类型数据。如语句int*p;p=1000;不合法;(内存空间的实际分配由操作系统负责)。指针变量的定义与初始化指针变量的定义指针变量的初始化一般形式:数据类型*指针名=初始地址值;赋给指针变量,不是赋给目标变量例inti;int*p=&i;变量必须已定义过类型应一致例int*p=&i;inti;例inti;int*p=&i;int*q=p;用已初始化指针变量作初值例main(){inti;staticint*p=&i;..............}()不能用auto变量的地址去初始化static型指针指针变量的引用&与*运算符含义含义:取变量的地址单目运算符优先级:2结合性:自右向左含义:取指针所指向变量的内容单目运算符优先级:2结合性:自右向左两者关系:互为逆运算理解…...…...2000200420062005整型变量i10变量i_pointer2001200220032000指针变量i_pointer-----指针变量,它的内容是地址量*i_pointer----指针的目标变量,它的内容是数据&i_pointer---指针变量占用内存的地址200010i_pointer*i_pointer&i_pointerii_pointer&i&(*i_pointer)i*i_pointer*(&i)i_pointer=&i=&(*i_pointer)i=*i_pointer=*(&i)直接访问与间接访问直接访问:按变量名称存取变量值间接访问:通过存放变量地址的变量去访问变量例i=3;-----直接访问指针变量…...…...2000200420062005整型变量i10变量i_pointer20012002200320003例*i_pointer=20;-----间接访问20指针变量…...…...2000200420062005整型变量i10变量i_pointer2001200220032000整型变量k例k=i;--直接访问k=*i_pointer;--间接访问10例k=i;k=*i_pointer;例8.1取地址运算符&和指针运算符*的简单使用。#includestdio.hvoidmain(){inta=100,*pa;pa=&a;//把变量a的地址赋给paprintf(a=%d,*pa=%d\n,a,*pa);a=200;printf(a=%d,*pa=%d\n,a,*pa);*pa=300;//给pa的对象赋值,即给a赋值printf(a=%d,*pa=%d\n,a,*pa);}程序运行:a=100,*pa=100a=200,*pa=200a=300,*pa=3001002000:a?pa2000200300例8.2使两个指针变量交换指向。#includestdio.hvoidmain(){inta=10,b=20;int*p1=&a,*p2=&b,*p;printf(a=%d,b=%d,*p1=%d,*p2=%d\n,a,b,*p1,*p2);p=p1;p1=p2;p2=p;printf(a=%d,b=%d,*p1=%d,*p2=%d\n,a,b,*p1,*p2);}程序运行:a=10,b=20,*p1=10,*p2=20a=10,b=20,*p1=20,*p2=10101001:a1001p1201012:b1012p210011012例l8-2d1使两指针变量所指对象的值交换main(){int*p1,*p2,a,b,t;a=10;b=20;p1=&a;p2=&b;t=*p1;*p1=*p2;*p2=t;printf("a=%d,b=%d\n",a,b);}10ap120bp2运行结果:a=20,b=10t201010例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随机零指针与空类型指针零指针:(空指针)定义:指针变量值为零表示:int*p=0;p指向地址为0的单元,系统保证该单元不作它用表示指针变量值没有意义#defineNULL0int*p=NULL:p=NULL与未对p赋值不同用途:避免指针变量的非法引用在程序中常作为状态比较例int*p;......while(p!=NULL){...…}void*类型指针表示:void*p;使用时要进行强制类型转换例char*p1;void*p2;p1=(char*)p2;p2=(void*)p1;表示不指定p是指向哪一种类型数据的指针变量8.1.3指针的运算指针变量的赋值运算inta,array[10],i=2,*p1,*p2;p1=&a;(将变量a地址p)p1=array;(将数组array首地址p)p1=&array[i];(将数组元素地址p)p1=p2;(指针变量p2值p1)不能把一个整数p,也不能把p的值整型变量如inti,*p;p=1000;()i=p;()指针变量与其指向的变量具有相同数据类型指针的算术运算:pipid(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+91指针变量的关系运算若p1和p2指向同一数组,则p1p2表示p1指的元素在前p1p2表示p1指的元素在后p1==p2表示p1与p2指向同一元素若p1与p2不指向同一数组,比较无意义p==NULL或p!=NULL间接访问主调函数中的变量——地址传递特点:共享内存,“双向”传递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);}例将两数从大到小输出…...…...20002008200A2002200420065变量a变量b(main)9变量temp变量y变量x(swap)55959COPY8.1.4指针作为函数的参数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);}例l8-3d1将两数从大到小输出值传递…...…...20002008200A2002200420065变量a变量b(main)9运行结果:5,9swap1(int*p1,int*p2){intp;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)swap1(pointer_1,pointer_2);printf(\n%d,%d\n,a,b);}…...20002008200A200220042006200C200E2010...59整型变量a整型变量b(main)指针pointer_1指针pointer_220002002(swap1)指针p1指针p2整型p5920002002COPY5例8.3编写函数swap1,交换两个变量的值。swap1(int*p1,int*p2){intp;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)swap1(pointer_1,pointer_2);printf(\n%d,%d\n,a,b);}…...20002008200A200220042006200C200E2010...59整型变量a整型变量b(main)指针pointer_1指针pointer_22000200259例8-3编写函数swap1,交换两个变量的值。运行结果:9,5地址传