第6章 指针教学课件PPT.ppt
《第6章 指针教学课件PPT.ppt》由会员分享,可在线阅读,更多相关《第6章 指针教学课件PPT.ppt(79页珍藏版)》请在淘文阁 - 分享文档赚钱的网站上搜索。
1、完整版教学课件完整版教学课件第6章 指针 第6章 指针u6.1 寻觅芳踪:初识指针寻觅芳踪:初识指针u6.2 强强联合:指针和函数强强联合:指针和函数u6.3 灵活高效:指针和数组灵活高效:指针和数组u6.4 本章小结本章小结屠龙刀倚天剑强转与指针,并称C语言的两大神器用好了可呼风唤雨,威力无比用不好也会伤及自身 6.1 初识指针初识指针n理解指针要从变量的地址谈起n是谁惹的祸?n几乎全是由指针和数组引起的非法内存访问导致的n黑客攻击服务器利用的bug绝大部分都是指针和数组造成的 6.1 初识指针初识指针6.1变量的内存地址变量的内存地址变量的地址变量的地址(Address)变量名变量名内存:
2、计算机内的存储部件内存:计算机内的存储部件所有指令和数据都保存在内存里所有指令和数据都保存在内存里速度快,可随机访问,但掉电即失速度快,可随机访问,但掉电即失编译或函数调用时为变量分配内存单元编译或函数调用时为变量分配内存单元0000ContentsContentsContentsContentsContentsContentsContents变量的值变量的值某存储区域6.1.1 内存地址和指针内存地址和指针 6.1 初识指针初识指针 6.1变量的内存地址变量的内存地址变量的地址变量的地址内存中的每个字节都有唯一的编号(地址)内存中的每个字节都有唯一的编号(地址)地址是一个十六进制无符号整数地
3、址是一个十六进制无符号整数其字长一般与主机相同其字长一般与主机相同地址按字节编号,按类型分配空间地址按字节编号,按类型分配空间0000ContentsContentsContentsContentsContentsContentsContents某存储区域Address Operator 6.1 初识指针初识指针如何读写内存中的数据?如何读写内存中的数据?0000ContentsContentsContentsContentsContentsContentsContents某存储区域只要指明要访问的变量的内存单元地址只要指明要访问的变量的内存单元地址就可以立即访问到变量所在的存储单元就可以立即
4、访问到变量所在的存储单元scanf(%d, &a); 6.1 初识指针初识指针【例例】使用取地址运算符使用取地址运算符& &取出变量的地取出变量的地址,然后将其显示在屏幕上。址,然后将其显示在屏幕上。表示输出变量表示输出变量a的地址值的地址值 6.1 初识指针初识指针如何读写内存中的数据?如何读写内存中的数据?0000ContentsContentsContentsContentsContentsContentsContents某存储区域直接寻址:直接寻址:按变量地址存取变量值按变量地址存取变量值scanf(%d, &a); 6.1 初识指针初识指针 6.1 初识指针初识指针如何读写内存中的数
5、据?如何读写内存中的数据?0000ContentsContentsContentsContentsContentsContents0 x0037b000某存储区域间接寻址:间接寻址:通过存放变量地址的变量去访问变量通过存放变量地址的变量去访问变量n 存放变量的地址需要一种特殊类型的变量存放变量的地址需要一种特殊类型的变量指针指针(Pointer)类型类型n 指针变量指针变量具有指针类型的变量具有指针类型的变量 指针变量指针变量 指向指向 变量变量变量的地址(指针)变量的地址(指针)变量值变量值变量地址存入变量地址存入指针变量指针变量 6.1 初识指针初识指针指针变量的定义v一般形式: 存储类型
6、 基类型类型 * 指针变量名;合法标识符指针变量本身的存储类型指针所指变量(目标变量)的数据类型例 int * p1,* p2; float * q ; static char * name;注意:注意: 区别区别int int * * p1, p1,* * p2; p2;与与int int * * p1,p2; p1,p2; 指针变量名是指针变量名是p1,p2 ,p1,p2 ,不是不是* *p1,p1,* *p2p2 指针变量只能指向定义时所规定类型的变量指针变量只能指向定义时所规定类型的变量 指针变量定义后,变量值不确定,应用前必须先赋值指针变量定义后,变量值不确定,应用前必须先赋值。 6
7、.1 初识指针初识指针6.1.2 指针变量的定义、初始化与引用指针变量的定义、初始化与引用 指针变量的初始化指针变量的初始化一般形式:一般形式:存储类型存储类型 数据类型数据类型 * 指针名指针名=初始地址值初始地址值;例 int i; int * p=&i;类型的一致性例 int i; int * p=&i; int * q=p;用已初始化指针变量作初值 6.1 初识指针初识指针int i,* p;p=&i; int * p;float * q;p=q;int i;float * p;p=&i;int * p;p=100;指针变量只指针变量只存放地址存放地址!一个指针变量不能指向与一个指针变
8、量不能指向与其类型不同的变量其类型不同的变量!我是真的,我是真的,你猜对了吗?你猜对了吗?应在类型相同的应在类型相同的指针变量之间赋指针变量之间赋值!值! 【案例【案例6.1】 打印普通变量的地址值和指针变量的值。打印普通变量的地址值和指针变量的值。#include int main()()int x, y;int * pX=&x; /*定义指针变量定义指针变量pX,并将其赋值为变量,并将其赋值为变量x的地址的地址*/int * pY, * pZ; /*定义指针变量定义指针变量pY与与pZ*/pY=&y; /*将变量将变量y的地址赋给指针变量的地址赋给指针变量pY*/ pZ=pY; /*将指针
9、将指针pY的值赋给指针的值赋给指针pZ,让,让pZ指向指向pY,二者指向同一内存空间,二者指向同一内存空间*/printf( (&x=%p, pX=%pn,&x,pX) );/*打印变量打印变量x的地址和指针变量的地址和指针变量pX的值的值*/printf( (&y=%p, pY=%pn,&y,pY) );/*打印变量打印变量y的地址和指针变量的地址和指针变量pY的值的值*/ printf( (pZ=%pn,pZ) ); /*打印指针变量打印指针变量pZ的值的值*/ return 0;变量变量x的地址和指针变量的地址和指针变量pX的值相等,变的值相等,变量量y的地址和指针变量的地址和指针变量p
10、Y的值也相等,这的值也相等,这也证明了指针变量也证明了指针变量pX指向变量指向变量x,指针变,指针变量量pY指向变量指向变量y。而指针变量。而指针变量pZ与与pY指指向同一个内存空间,由此二者的地址值向同一个内存空间,由此二者的地址值相等。相等。 6.1 初识指针初识指针零指针与空类型指针v零指针(空指针): 指针变量值为零l表示: int * p=0; p指向地址为0的单元,系统保证该单元不作它用表示指针变量值没有意义#define NULL 0int * p=NULL;lp=NULL与未对p赋值不同l用途: u避免指针变量的非法引用u在程序中常作为状态比较 例 int * p; . whi
11、le(p!=NULL) . vvoid *类型指针l表示: void * p; l使用时要进行强制类型转换例 char * p1; void * p2; p1=(char *)p2; p2=(void *)p1;表示不指定p是指向哪一种类型数据的指针变量 6.1 初识指针初识指针访问指针变量访问指针变量所所指向指向的的变量,只需在指针变量前加一个变量,只需在指针变量前加一个“*”(取值运算符取值运算符) ),其,其格式格式为为:*指针表达式指针表达式;例如:例如:int i=0;/*定义并初始化变量定义并初始化变量i*/int * p=&i; /*定义指针变量定义指针变量p,将变量,将变量i的
12、地址赋给变量的地址赋给变量p,此时,此时p指向变量指向变量i*/printf( (%d, *p) ); /*打印指针变量打印指针变量p所指向内存空间中存放的值,也就是打印所指向内存空间中存放的值,也就是打印i的值的值*/此时的此时的*p就相当于就相当于i,如果再有一句赋值,如果再有一句赋值“*p=1;”就相当于完成赋值就相当于完成赋值“i=1; ”。运算符运算符&( (取地址运算符取地址运算符) )和和“*”互为逆运算,很显然,若定义指针变量互为逆运算,很显然,若定义指针变量p,则,则&( (*p) )与与p是等价的。是等价的。 6.1 初识指针初识指针例 int main( ) int i=
13、10; int * p; *p=i; printf(“%d”,*p); return 0; 危险!危险!例 int main( ) int i=10,k; int * p; p=&k; *p=i; printf(“%d”,*p); return 0; 指针变量必须先赋值,再使用.2000200420062005整型变量整型变量i10指针变量指针变量p200120022003随机随机 6.1 初识指针初识指针6.1.3 6.1.3 指针变量的移动和比较指针变量的移动和比较指针变量可以加上或减去一个整数指针变量可以加上或减去一个整数( (常量或变量常量或变量) ),达到改变地址值的目的。,达到改变
14、地址值的目的。在这一过程中,可以认为是指针变量发生了移动。在这一过程中,可以认为是指针变量发生了移动。若若p是一个指针变量,则是一个指针变量,则p+1表示指向下一个内存空间。需要注意的是,执行表示指向下一个内存空间。需要注意的是,执行p+1时,并不是将时,并不是将p的值的值( (地址地址) )进行简单的加进行简单的加1,而是加上,而是加上p的基类型占用的字节的基类型占用的字节数。数。例如,如果例如,如果p的基类型是的基类型是int类型,占类型,占4字节,则字节,则p+1意味着意味着p的值加的值加4字节,字节,p-1意味着意味着p的值减的值减4字节。变量字节。变量a的地址是的地址是0001,p的
15、值也是的值也是0001,当执行,当执行“p = p+1”时,由于时,由于p的基类型是的基类型是int型,在内存中占型,在内存中占4字节,因此执行后,字节,因此执行后,p就指向了就指向了“0001+4字节字节”后面的位置,即地址后面的位置,即地址0005的位置。的位置。若若y是整数,是整数,p是一个指针变量,如是一个指针变量,如果它所指向的数据类型在内存中占据果它所指向的数据类型在内存中占据x字节的存储空间,则字节的存储空间,则p+y表示在表示在p的地的地址值基础上址值基础上向后移动向后移动 x*y字节。反之,字节。反之,p-y表示在表示在p的地址值基础上的地址值基础上向前移动向前移动 x*y字
16、节。因此,不同基类型的指针变字节。因此,不同基类型的指针变量量p,在同样执行,在同样执行p+1后,其结果也是后,其结果也是不同的。不同的。 6.1 初识指针初识指针指针的自增指针的自增/自减运算类似于普通变量的自增自减运算。例如自减运算类似于普通变量的自增自减运算。例如“p+”与与“+p”就相当于就相当于“p=p+1”,“-p”与与“p-”相当于相当于“p=p-1”。此外,结合运算优先级,。此外,结合运算优先级,*p+等价于等价于*( (p+) ),作用是先得到,作用是先得到p所指向的变量的值所指向的变量的值( (即即*p) ),然后使,然后使p加加1。要注。要注意的是,它与意的是,它与*+p
17、不同,不同,*+p等价于等价于*( (+p) ),它是先使,它是先使p加加1,再取得,再取得*p的值。类似的值。类似地,地,*( (p-) )与与*( (-p) )的含义也是不同的。的含义也是不同的。【案例【案例6.2】 利用指针实现变量值的变化。利用指针实现变量值的变化。#include int main()() int a=5,* p; /*定义指针变量定义指针变量p*/ p=&a; /*让指针让指针p指向指向a*/ printf( (a=%d,*p=%dn,a,*p) ); /*输出输出a的值和的值和p所指向变量的值所指向变量的值*/ *p=8; /*对对p所指向的变量赋值,就相当于对所
18、指向的变量赋值,就相当于对a赋值赋值*/ printf( (a=%d,*p=%dn,a,*p) ); printf( (Enter a:) ); scanf( (%d,&a) ); printf( (a=%d,*p=%dn,a,*p) ); ( (*p) )+; /*将将p所指向的变量值加所指向的变量值加1,也就是将,也就是将a的值加的值加1*/ printf( (a=%d,*p=%dn,a,*p) ); return 0; 6.1 初识指针初识指针 如果是同类的指针进行相减运算,其结果是两个指针的地址之差除以指针基如果是同类的指针进行相减运算,其结果是两个指针的地址之差除以指针基类型所占字节
19、数。例如,整型指针变量类型所占字节数。例如,整型指针变量p1指向的变量的地址为指向的变量的地址为20000,整型指针,整型指针变量变量p2 指向的变量的地址为指向的变量的地址为20016,那么,那么,p2-p1的结果是的结果是“(20016-20000) )/4=4”,表示指针变量,表示指针变量p2所指的元素与所指的元素与p1所指的元素之间相隔所指的元素之间相隔4个同类型个同类型元素。元素。 既然指针可以移动,也就相应地存在大小比较,存在着关系运算。常用的关既然指针可以移动,也就相应地存在大小比较,存在着关系运算。常用的关系运算符系运算符,如如“=”、“!=”、“”、“=”等都适用于指等都适用
20、于指针。针。 在实际编程时,对单独零散的变量的指针进行加减运算和比较的意义不是太在实际编程时,对单独零散的变量的指针进行加减运算和比较的意义不是太大,但与批量数据的处理大,但与批量数据的处理(如数组等如数组等)结合使用时,指针这些的运算就有很大的结合使用时,指针这些的运算就有很大的作用了作用了。 注意:注意:只有同类指针之间才可以进行关系运算,对两个毫无关联的指针比较只有同类指针之间才可以进行关系运算,对两个毫无关联的指针比较大小是没有意义的,如指向两个毫不相干内存空间大小是没有意义的,如指向两个毫不相干内存空间(如两个不同的数组如两个不同的数组)的指针的指针之间无法比较大小。此外,不能对同类
21、指针进行相加运算,若存在指针变量之间无法比较大小。此外,不能对同类指针进行相加运算,若存在指针变量p1与与p2,则,则p1+p2没有任何意义。没有任何意义。 6.1 初识指针初识指针 6.2 指针与函数指针与函数6.2.1 6.2.1 指针变量作为函数参数指针变量作为函数参数【案例【案例6.3】 有张三和李四两位小朋友,张三手上拿着有张三和李四两位小朋友,张三手上拿着“书书”,李四手上,李四手上拿着拿着“画画”,由于张三非常喜欢李四的画,而李四也十分喜欢张三的书,因,由于张三非常喜欢李四的画,而李四也十分喜欢张三的书,因此他们决定此他们决定 “换一换换一换”,试利用程序实现这一交换的场景。,试
22、利用程序实现这一交换的场景。#define BOOK 0 /*定义书为定义书为0*/#define PICTURE 1 /*定义画为定义画为1*/#include swap( (int x, int y) ) int temp; temp=x; x=y; y=temp;int main()() int a=BOOK, b=PICTURE; swap( (a,b) ); if(a=PICTURE) )&( (b=BOOK) printf( (SUCCESSn) ); else printf( (FAILUREn) ); return 0;普通变量作为函数参数普通变量作为函数参数【案例【案例6.3
23、 改造改造1】 利用利用指针变量做指针变量做函数函数参数参数实现交换。实现交换。#include #define BOOK 0 /*定义书为定义书为0*/#define PICTURE 1 /*定义画为定义画为1*/swap( (int * x, int * y) ) int temp; temp=*x; *x=*y; *y=temp;int main()() int a=BOOK, b=PICTURE; swap( (&a,&b) ); if(a=PICTURE) )&( (b=BOOK) printf( (SUCCESSn) ); else printf( (FAILUREn) ); re
24、turn 0; 6.2 指针与函数指针与函数【案例【案例6.3 改造改造2】有风险的交换代码。有风险的交换代码。swap( (int *x, int *y) ) int * temp; *temp=*x; *x=*y; *y=*temp; 语句语句“int * temp;”后没有立即对指针变量后没有立即对指针变量temp初始化,它可能指向系统初始化,它可能指向系统重要数据的区域,而后马上执行一句赋值重要数据的区域,而后马上执行一句赋值“*temp=*x;”,就会将系统重要,就会将系统重要区域的值覆盖,从而导致系统崩溃。因此,这种对未知单元写操作是很危险区域的值覆盖,从而导致系统崩溃。因此,这种
25、对未知单元写操作是很危险的。可将的。可将 “int * temp;”修改为修改为“int * temp, t; temp=&t;”,其中新定义的,其中新定义的t一定被分配在空闲的内存空间中,这就比较安全了。一定被分配在空闲的内存空间中,这就比较安全了。 因此因此,在进行指针的使用时,要时刻注意以下三大原则在进行指针的使用时,要时刻注意以下三大原则: (1)永远要清楚每个指针指向了哪里。永远要清楚每个指针指向了哪里。 (2)永远要明确指针所指向单元的内容是什么。永远要明确指针所指向单元的内容是什么。 (3)永远不要使用未初始化的指针变量。永远不要使用未初始化的指针变量。 6.2 指针与函数指针与
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- 第6章 指针教学课件PPT 指针 教学 课件 PPT
限制150内