C语言程序设计第八章指针.ppt
《C语言程序设计第八章指针.ppt》由会员分享,可在线阅读,更多相关《C语言程序设计第八章指针.ppt(106页珍藏版)》请在淘文阁 - 分享文档赚钱的网站上搜索。
1、第八章第八章 指针指针C8.1概述C程序设计中使用指针可以:使程序简洁、紧凑、高效有效表示复杂的数据结构动态分配内存方便使用字符串有效使用数组调用函数时可以得到多于1个的值直接处理内存地址C8.2指针的概念一、数据的存取一、数据的存取1、内存地址内存的每个字节都有一个编号,这个编号称为“内存地址”。低地址高地址2000200220043000程序中的每个数据都对应着内存中的一个地址,从该地址开始的一个或多个字节用来存放该数据。inti,j,k;ijk2、内存单元的地址和内存单元的内容的区别:若i=3,j=5;00000011000000000000010100000000从图中可以看出它们的区
2、别。C程序编译后已经没有i、j、k这些变量名了,而是将变量名转换为变量的地址,计算机通过内存地址对变量进行存取。二、直接访问和间接访问二、直接访问和间接访问1、直接访问方式:低地址高地址2000200220043000ijk00000011000000000000010100000000、i=3;、j=i+2;、printf(%d,i);、scanf(%d,&i);、k=i+j;如何执行?0000100000000000按变量的地址存取变量的方式2、间接访问方式:语言中可以定义整型变量、实型变量、字符型变量,各自存放相应类型的数据;另外还可以定义和使用一种特殊类型的变量,用来存放变量的地址用来
3、存放变量的地址用来存放变量的地址用来存放变量的地址。C低地址高地址2000200220043000ijk000000110000000000000101000000000000100000000000假设已经定义变量ip用来存放整型变量的地址,它被分配到内存单元3000和3001。ip通过执行语句:ip=&i;将整型变量i的地址存放到变量ip中,即ip的值为变量i所对应的内存单元的起始地址2000。2000现在要存取i的值可以这样做:先找到存放i的地址的内存单元地址(3000和3001),从中取出变量i的地址(2000),然后再对2000和2001单元进行存取这就称为间接访问。C通过变量ip知
4、道变量i的地址,从而找到变量i的内存单元,因此说变量ip指向变量i,用箭头表示这种“指向”关系。20003000ip2000i变量ip的值为2000,即变量i的地址,这样就在ip和i之间建立了一种联系:3、关于“指向”的含义:C4、为了表示将数值10送到变量i中,可以有两种表示方法:、将10送到变量i所标志的单元中;2000i10、将10送到变量ip所指向的单元中;20003000ip2000i10C三、指针与指针变量通过地址能找到所需的变量单元,可以说:地址“指向”该变量单元。因此,把一个变量的地址称为该变量的“指针”。低地址高地址2000200220043000ijk00000011000
5、0000000000101000000000000100000000000ip20002000如果一个变量专门用来存放另一个变量的地址,则称它为“指针变量”。指针变量的值(即指针变量中存放的值)是指针(地址)。注意区分“指针”和“指针变量”这两个概念。C.2000200420062005整型变量i10变量i_pointer200120022003指针与指针变量指针:一个变量的地址指针变量:专门存放变量地址的变量2000指针指针变量 变量的内容 变量的地址指针变量变量变量地址(指针)变量值 指向地址存入指针变量C、数据所占有的内存单元个数是由其数据类型决定的;、首地址:即第一个单元的地址;、表示
6、地址的数与整数的区别;、变量i、j的地址可能相邻,也可能不相邻,是由系统分配的,我们不必关心。、程序中定义的每个变量在编译后都占有各自的内存单元,系统是通过内存地址对变量进行存取的;四、说明四、说明C8.3指针变量的定义和引用指针运算符:*为了表示指针变量和它所指向的变量之间的联系,用“*”表示指向的关系。如:ip代表指针变量,*ip表示ip所指向的变量。即*ip也代表一个变量。例:、若:ip=&i;i=5;*ip=5;结论:*ip与i等价。、x=i+1;x=*ip+1;C取地址运算符:&它与一个变量连用,以得到该变量的内存地址。说明:、取地址运算符只能作用于变量:&x,&a5;不能作用于常量
7、、表达式或寄存器变量:&25,&(i+1);、不能把整数赋值给一个指针变量:ip=3000;、不能把一个指针变量的值赋值给一个整型变量:x=ip;C&与*运算符含义含义:取变量的地址单目运算符优先级:2结合性:自右向左含义:取指针所指向变量的内容单目运算符优先级:2结合性:自右向左两者关系:互为逆运算理解.2000200420062005整型变量i10变量i_pointer2001200220032000指针变量i_pointer-指针变量,它的内容是地址量*i_pointer-指针的目标变量,它的内容是数据&i_pointer-指针变量占用内存的地址200010i_pointer*i_poi
8、nter&i_pointerii_pointer&i&(*i_pointer)i*i_pointer*(&i)i_pointer=&i=&(*i_pointer)i=*i_pointer=*(&i)C8.3.1指针变量的定义与赋值一、指针变量的定义合法标识符指针变量本身的存储类型指针的目标变量的数据类型表示定义指针变量不是*运算符例int*p1,*p2;float*q;staticchar*name;存储类型 数据类型 *指针变量名;C说明:“*”只表示定义的变量为指针变量,但指针变量名中并不包含“*”;*是指针变量的标志,不可丢掉;指针变量定义时,指定了它所指向的变量的数据类型;指针变量定义
9、时必须指定其所指向的变量的数据类型,而且使用过程中只能指向同一种类型的变量。指针变量定义后,系统为变量分配一个存储单元,用来存放地址;根据存储单元的长度分为大存储模式(长指针,4Byte)和小存储模式(短指针,2Byte);指针变量定义后,若不赋值,其值是不确定的。C二、指针变量的赋值1、赋值语句:inti,j,*p1,*p2;p1=&i;p2=&j;charch,*cp1,*cp2;cp1=&ch;cp2=&ch;2、初始化:int i,*p1=&i;int i,*p1;int i,*p1;p1=&i;p1=&i;C一般形式:存储类型 数据类型*指针变量名=初始地址值;赋给指针变量,不是赋给
10、目标变量例 int i;int *p=&i;变量必须已说明过类型应一致例 int i;int *p=&i;int *q=p;用已初始化指针变量作初值例 main()int i;static int *p=&i;.()不能用auto变量的地址去初始化static型指针C3、说明:、指针变量定义后,若不赋值,其值是不确定的;、可以给指针变量赋空值(NULL),使指针变量不指向不指向不指向不指向任何变量任何变量任何变量任何变量;ip=NULL;ip=NULL;#define NULL 0#define NULL 0、指针变量的值为空值(NULL)与未对指针变量赋值,意义不同;、只能是同类型变量的地址
11、进行赋值;int i,*ip;int i,*ip;char ch,*cp;char ch,*cp;ip=&i;ip=&i;cp=&ch;cp=&ch;ip=&ch;ip=&ch;cp=&i;cp=&i;、可以将数组名或函数名赋给某些类型的指针变量;int a10,*ip;int a10,*ip;ip=&a;ip=&a;ip=a;ip=a;、不能将一个整型量(或任何其它非地址类型的数据)赋给一个指针变量;int *ip;int *ip;ip=3000;ip=3000;C8.3.2指针变量的引用int a,*p1,*p2;p1=&a;printf(%x,p1);p2=p1;*p1=3;printf
12、(%d,*p1);int a,*p1,*p2;不定a不定p1不定p2p1=&a;&aprintf(%x,p1);p2=p1;&a*p1=3;3printf(%d,*p1);C1、两个运算符:&和*&:取地址运算符;*:指针运算符(间接访问符);2、说明:、&既可作用于一般变量,也可作用于指针变量;、*只能作用于指针变量;、表达式中的*p与变量定义中的*p含义不同;inti,*p=&i;*p=5;、inta,*p=&a;inta,*p;p=&a;inta,*p;*p=&a;3、指针变量可以进行的操作:inta,*p1,*p2;、赋值:p1=&a;p2=p1;、输出:printf(%x,p1);、
13、取内容:*p1=5;a=5;printf(%d,*p1);C例8.1#include void main()int a1=11,a2=22;int *p1,*p2;p1=&a1;p2=&a2;printf(%d,%dn,*p1,*p2);p2=p1;printf(%d,%dn,*p1,*p2);11a122a2不定p1不定p2&a1&a2*p1*p2&a1*p2*p2多个指针可以指向同一个存储单元。但在某一时刻,一个指针变量只能指向一个存储单元,因为指针变量在某一时刻只能存放一个变量的地址值。C例8.2#include void main()int a1=11,a2=22;int *p1,*p
14、2,*p;p1=&a1;p2=&a2;printf(%d,%dn,*p1,*p2);p=p1;p1=p2;p2=p;printf(%d,%dn,*p1,*p2);11a122a2不定p1不定p2不定p&a1&a2*p1*p2&a1&a2*p1*p1&a1*p2*p2使两个指针交换指向使两个指针交换指向C例8.3#include void main()int a1=11,a2=22,t;int *p1,*p2;p1=&a1;p2=&a2;printf(%d,%dn,a1,a2);t=*p1;*p1=*p2;*p2=t;printf(%d,%dn,a1,a2);11a122a2不定t不定p1不定p
15、2&a1*p1&a2*p2112211交换两个指针变量所指向变量的值交换两个指针变量所指向变量的值C 反映了指针变量的引用方法:反映了指针变量的引用方法:、将变量的地址赋给指针变量(p1=&a1)、将一个指针变量赋给另一个指针变量(p2=p1)、通过指针变量间接访问它所指向的变量(*p1)4、*和&运算符的进一步说明:、若有:p1=&a;则*p1等价于a;&*p1&(*p1)&a、*&a*(&a)*pa、(*p)+等价于:a+不同于:*p+*p+*(p+)C指针作为函数参数输入两个整数a、b,按大小顺序输出。#include void swap(int x,int y)int temp;tem
16、p=x;x=y;y=temp;void main()int a,b;printf(nInput a,b:);scanf(%d%d,&a,&b);if (ab)swap(a,b);printf(max=%d,min=%dn,a,b);注意:语言中的函数调用采用“传值”方式,即单向传递方式。即:主调函数可以将实参的值传递给被调函数的形参,但不能通过改变形参的值而改变实参的值。Eg804.cppC#include void swap(int*px,int*py)int temp;temp=*px;*px=*py;*py=temp;void main()int a,b,*p1,*p2;printf(n
17、Input a,b:);scanf(%d%d,&a,&b);p1=&a;p2=&b;if (ab)swap(p1,p2);printf(max=%d,min=%dn,a,b);5a8b&ap1&bp2不定px不定py&a&b858a5b&ap1&bp2C#include void swap(int*px,int*py)int*temp;*temp=*px;*px=*py;*py=*temp;void main()int a,b,*p1,*p2;printf(nInput a,b:);scanf(%d%d,&a,&b);p1=&a;p2=&b;if (ab)swap(p1,p2);printf(
18、max=%d,min=%dn,a,b);*temp是指针变量temp所指向的变量,但temp中并无确定的地址值,其值不确定;*temp所指向的单元也不确定。因此,对*temp赋值可能会破坏系统的正常工作状况。应该将*px的值赋给一个整型变量,用整型变量作为临时存储空间实现*px和*py的交换。C#include void swap(int*px,int*py)int*p;p=px;px=py;py=p;void main()int a,b,*p1,*p2;printf(nInput a,b:);scanf(%d%d,&a,&b);p1=&a;p2=&b;if (ab)swap(p1,p2);p
19、rintf(max=%d,min=%dn,a,b);5a8b&ap1&bp2不定px不定py&a&b&b&a5a8b&ap1&bp2C如果想通过函数调用得到几个要改变的值,可以:、在主调函数中设n个变量;、将n变量的地址作为实参传给所调用的函数的形参;、通过形参指针变量,改变该n个变量的值;、主调函数就可以使用这些改变了的值;C8.4指针与数组一个变量有地址,称变量的地址为该变量的指针指针;指针变量可以指向变量,当然也可以指向数组和数组元素;每个数组都有一个起始地址,数组的起始地址称为数组的指针数组的指针;一个数组包含若干元素,每个数组元素都在内存中占用一定的存储单元,即都有相应的地址,数组元
20、素的地址称为数组元素的指针数组元素的指针;数组元素的指针变量数组元素的指针变量就是专门用来存放数组元素地址的变量。C指针与数组指针可以指向数组和数组元素当一个指针指向数组后,对数组元素的访问,既可以使用数组下标,也可以使用指针用指针访问数组元素,程序的效率更高(占内存少,运行速度快)用下标访问数组元素程序更清晰C8.4.1指向数组元素的指针变量定义类型名*指针变量名例:inta10;/*定义包含10个整型元素的数组a*/int*p;/*定义指向整型变量的指针变量p*/指向数组元素的指针变量,其类型应与数组元素相同为了让指针p指向数组a,应把数组a的地址赋给指针变量pCa0a1a2a3a4a5a
21、6a7a8a9a a不定p赋值:p=&a0;&a0说明:、语言中的数组名代表数组首地址,即第0号元素的地址;p=&a0;p=a;、定义时可以进行初始化:int*p=&a0;int*p;p=&a0;int*p;*p=&a0;int*p=a;int*p=a;C8.4.2通过指针访问一维数组inta10;int*p;如果:p=a;(或p=&a0;)则*p=5;a0=5;p=&a3;*p=5;a3=5;又如果:当指针p指向数组a后,可用指针p访问数组的各个元素。C通过指针访问一维数组C规定:若指针变量p已指向数组中的一个元素,则p1指向该数组中的下一个元素如果指针p指向数组a(指向数组第一个元素a0)
22、,则:p+1指向下一个元素a1,注意不是将p值简单加1。如果数组元素是整型,p+1表示p的地址加2;如果数组元素是实型,p+1表示p的地址加4;如果数组元素是字符型,p+1表示p的地址加1C通过指针访问一维数组p+i指 向 元 素 ai。可 以 使 用*(p+i)访问元素aip+i也可以记作a+i。指向元素ai。指向数组的指针变量也可以带下标,如,pi与*(p+i)等价,表示元素ai。C指针的运算指针变量的赋值运算p=&a;(将变量a地址p)p=array;(将数组array首地址p)p=&arrayi;(将数组元素地址p)p1=p2;(指针变量p2值p1)不能把一个整数p,也不能把p的值整型
23、变量C指针的运算指针的算术运算pipid(i为整型数,d为p指向的变量所占字节数)p+,p-,p+i,p-i,p+=i,p-=i等若p1与p2指向同一数组,p1-p2=两指针间元素个数(p1-p2)/dp1+p2无意义C指针的运算例p指向float数,则p+1p+14例p指向int型数组,且p=&a0;则p+1指向a1例inta10;int*p=&a2;p+;*p=1;例inta10;int*p1=&a2;int*p2=&a5;则:p2-p1=3;a0a1a2a3a4a5a6a7a8a9a数组pp+1,a+1p+i,a+ip+9,a+91C指针的运算指针变量的关系运算若p1和p2指向同一数组,
24、则p1p2表示p1指的元素在后p1=p2表示p1与p2指向同一元素若p1与p2不指向同一数组,比较无意义p=NULL或p!=NULLC数组元素表示方法a0a1a2a3a9.aa+9a+1a+2地址元素下标法a0a1a2a9a0a1a2a3a9.pp+9p+1p+2地址元素指针法*p*(p+1)*(p+2)*(p+9)变址运算符ai *(a+i)aipi*(p+i)*(a+i)*a*(a+1)*(a+2)*(a+9)p0p1p2p9C通过指针访问一维数组例10.1输出有10个元素的整型数组a中的全部元素1、下标法(常用,很直观)main()inta10;inti;for(i=0;i10;i+)s
25、canf(%d,&ai);printf(n);for(i=0;i10;i+)printf(%d,ai);例10.1输出有10个元素的整型数组a中的全部元素1、下标法(常用,很直观)main()inta10;inti;for(i=0;i10;i+)scanf(%d,&ai);printf(n);for(i=0;i10;i+)printf(%d,ai);下标C通过指针访问一维数组2、用数组名计算数组元素的地址。(效率与下标法相同,不常用)main()inta10;inti;for(i=0;i10;i+)scanf(%d,&ai);printf(n);for(i=0;i10;i+)printf(%d
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- 语言程序设计 第八 指针
限制150内