《C语言指针含例子.pptx》由会员分享,可在线阅读,更多相关《C语言指针含例子.pptx(70页珍藏版)》请在taowenge.com淘文阁网|工程机械CAD图纸|机械工程制图|CAD装配图下载|SolidWorks_CaTia_CAD_UG_PROE_设计图分享下载上搜索。
1、指针的用途指针在C 语言里应用极为广泛,是C语言的灵魂 使程序简洁、紧凑、高效使程序简洁、紧凑、高效 函数调用时批量参数的传递函数调用时批量参数的传递 内存的动态分配内存的动态分配 内存地址的直接处理内存地址的直接处理 复杂数据结构的简洁表达复杂数据结构的简洁表达等许多其它数据类型难以实现的复杂操作第1页/共70页培训目的掌握指针的概念、指针变量定义格式和引用掌握指针的运算方法指针应用举例第2页/共70页理解指针 指针 就是内存中的变量的地址第3页/共70页理解指针内存单元的编号:计算机内存单元的编号:计算机内存中,每一个字节的存内存中,每一个字节的存储单元都有一个编号(亦储单元都有一个编号(
2、亦称为称为内存地址内存地址)在在VC中,内存单元的编号中,内存单元的编号是一个是一个8位的十六进制数位的十六进制数(表示(表示32位二进制数,位二进制数,VC是是32位编译系统)位编译系统)如:如:0013FF7C0013FF7C0013FF7B0013FF7A0013FF790013FF00.第4页/共70页理解指针变量:实质是代表了“内存中的某个存储单元”。若在内存中定义了一个变量,则这个变量的内存的地址也就确定了。第5页/共70页理解指针变量的地址:表示变量在内存中的位置,其值是该变量在内存中所占存储单元的首字节的编号 第6页/共70页程序中:int i;float k;内存中每个字节有
3、一个编号-地址.2000200120022005内存02003ik 编译或函数调用时为其分配内存单元变量是对程序中数据存储空间的抽象理解指针第7页/共70页a在内存中占在内存中占4个字节,但个字节,但&a的值是的值是4个字节中首个字节的编号个字节中首个字节的编号 运算符运算符&:取地址,如:取地址,如&a 的值为变的值为变量量a的地址的地址 变量地址的输出可以用格式字符串变量地址的输出可以用格式字符串“%p”#include void main()int a;a=8;printf(%d%pn,a,&a);0013FF7C000000000013FF7B000000000013FF7A00000
4、0000013FF79000010000013FF00.理解指针第8页/共70页理解指针我们在程序中只需要指出变量名,无需知道每个变量在内存中的具体地址,每个变量与具体地址的联系是由C编译系统来完成的。对变量的存取操作就是对某个存储单元进行操作。这种直接按变量的地址存取变量值的方式称为“直接存取”。第9页/共70页 4502实验室二我有几种方法可以到实验室上机?2、按实验室所在的教室编号1、按实验室名字按变量名按变量地址 (即 指针)实验室一实验室三45034707理解指针第10页/共70页理解指针指针:也是一种变量,这种变量是用来存放内存地址的。第11页/共70页指针变量指针是 字节的编号,
5、即地址。指针也是要存储的,因此每个指针也需要一个变量来存储它。这个变量称为指针变量。指针变量的值是某一定义好的变量的地址。第12页/共70页理解指针变量&a2001p=&a;1011ap例:int*p;int a=3;101220020000001100000000间接存取:这种通过变量p到变量a的地址,然后再存取变量a的值的方式称为“间接存取”。指针变量p指向了变量a的含义是:指针p中存放了变量a的地址。第13页/共70页指针变量的定义定义形式如下:数据类型 *指针变量名;如:int *p1,num;表示的意思为:定义了一个指针变量p1,它指向的是一个整型数据。或者说,这时候p1中存储的是一
6、个整型变量的地址。注意:指针变量定义了之后,需要给它赋一个确定的变量的地址,否则,就会出现严重的系统错误。第14页/共70页指针变量的引用#include main()int a;/定义一个整型变量a int *p1;/定义指向整型变量的指针p1 p1=&a;/p1中应存放变量a的地址 *p1=5;/通过指针变量p1给a赋值 printf(a=%d,a);第15页/共70页例:int*p,*s,k=20;s=&k;p=&s;.2000200420062005整型变量k变量s200120022003变量p20072008202000200020042004s=*pk=*s所以:k=*p说明:(1
7、)指针p的基类型是int类型的指针。(2)是求地址运算符。(3)因为基本数据类型int、float、char等所需存储空间不同,所以需要定义指针变量的基类型。第16页/共70页例:指针的概念main()int a;int*pa=&a;a=10;printf(a:%dn,a);printf(*pa:%dn,*pa);printf(&a:%x(hex)n,&a);printf(pa:%x(hex)n,pa);printf(&pa:%x(hex)n,&pa);运行结果:a:10*pa:10&a:f86(hex)pa:f86(hex)&pa:f88(hex).f86f8af8cf8b整型变量a10指针
8、变量paf87f88f89f86第17页/共70页指针的运算方法指针的移动所谓移动指针就是对指针变量加上或减去一个整数,或通过赋值运算,使指针指向相邻的存储单元。只有当指针指向一串连续的存储单元时,还可以和指向同一串连续存储单元的指针进行相减的运算,除此之外,不可以对指针进行任何其它的算术运算。第18页/共70页指针的运算方法指针变量只能进行逻辑运算和相减的算术运算。不是指向同一数组的指针变量运算是无意义的。指针变量的值加1或减1,并不是地址加1中减1,而是加上或减去该变量在内存中所占的字节数,该字节数由指针的基类型决定。指针的移动第19页/共70页111122334455a0a1a2a3a4
9、pqDD00DD02DD04DD06DD08DD0ADD0CDD0EDD10ijDD12DD14DD16p=&a0;DD00q=p+2;DD04q+;DD06q-;DD04i=*p;11j=*q;33例:第20页/共70页指针的运算方法指针的比较在关系表达式中,可以对指针进行比较。即两个变量地址的比较。通常两个或多个指针指向同一目标时(一串连续的存储单元),比较才有意义。第21页/共70页指针应用举例-指针作为函数的参数指针可以作为参数在主调函数和被调用函数之间传递数据,通过指针可以在被调用函数中地调用中的变量进行引用,这也就使得通过形参改变对应实参的值有了可能,利用此形式就可以把两个或两个以
10、上的数据从被调用的函数返回到调用函数。第22页/共70页例例 将数从大到小输出将数从大到小输出指针应用举例-指针作为函数的参数swap(int x,int y)int temp;temp=x;x=y;y=temp;main()int a,b;scanf(%d,%d,&a,&b);if(ab)swap(a,b);printf(n%d,%dn,a,b);第23页/共70页指针应用举例-指针作为函数的参数.20002008200A2002200420065变量a 变量b(main)9 变量temp 变量y 变量x(swap)559 59COPYswap(int x,int y)int temp;te
11、mp=x;x=y;y=temp;main()int a,b;scanf(%d,%d,&a,&b);if(ab)swap(a,b);printf(n%d,%dn,a,b);第24页/共70页指针应用举例-指针作为函数的参数swap(int x,int y)int temp;temp=x;x=y;y=temp;main()int a,b;scanf(%d,%d,&a,&b);if(ab)swap(a,b);printf(n%d,%dn,a,b);值传递.20002008200A2002200420065变量a 变量b(main)9运行结果:5,9变量x 变量y(swap)变量t59595第25页/
12、共70页swap(int *p1,int *p2)int p;p=*p1;*p1=*p2;*p2=p;main()int a,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,%dn,a,b);.20002008200A200220042006200C200E2010.59整型变量a 整型变量b(main)指针pointer_1指针pointer_220002002(swap)指针p1指针p2整型p592000200
13、2COPY5运行结果:9,5地址传递通过传送地址值,可以在被调用函数中直接改变调用函数中的变量的值指针应用举例-指针作为函数的参数第26页/共70页指针应用举例-指向数组的指针变量例编写程序,定义一个含有15个元素的数组,并编写函数分别完成以下操作:(1)调用C库函数中的随机函数给所有元素赋以049之间的随机数;(2)输出数组元素中的值;(3)按顺序对每隔三个数求一个和数,并传回主函数;(4)最后输出所有求出的和值。第27页/共70页指针应用举例-指向数组的指针变量调用随机函数的方法如下:包含头文件n=rand()%x;n将得到一个在0(x-1)之间的随机数。第28页/共70页指针应用举例-指
14、向数组的指针变量分析:(1)由于要每隔3个数求一个和,15个数要求出5个和,所以主函数中要定义一个数组来存放5个和数。(2)两次输出元素的值,虽然输出对象不同,但是数组的类型相同,可调用一个函数来完成,只要指定不同的数组名,不同的输出个数即可。第29页/共70页指针应用举例-指向数组的指针变量#include stdio.h#include stdlib.h#define SIZE 15#define N 3void getrand(int*,int);void getave(int*,int*,int);void printarr(int*,int);第30页/共70页指针应用举例-指向数组
15、的指针变量void printarr(int*a,int n)int i;for(i=0;in;i+)printf(%4d,*(a+i);if(i+1)%5=0)printf(n);printf(n);第31页/共70页指针应用举例-指向数组的指针变量void getave(int*a,int*b,int n)int i,j=0,sum=0;for(i=0;in;i+)sum+=*(a+i);/*累加数组元素*/if(i+1)%3=0)/*每累加3个进行一次处理*bj=sum;sum=0;j+;第32页/共70页指针应用举例-指向数组的指针变量void getrand(int*a,int n)
16、int i;for(i=0;in;i+)*(a+i)=rand()%50;第33页/共70页指针应用举例-指向数组的指针变量main()int xSIZE,wSIZE/N=0;/*数组中置初值为0,准备存放5个数的和*/clrscr();getrand(x,SIZE);/*调用函数产生15个随机数放入数组x中*/printf(nOutput%d random numbers:n,SIZE);printarr(x,SIZE);/*输出15个随机数*/getave(x,w,SIZE);/*每3个数求一个和放入数组w中*/printf(nOutput 5 sum numbers:n);printar
17、r(w,SIZE/N);/*输出5个数的和*/第34页/共70页指针应用举例-指向数组的指针变量第35页/共70页指针应用举例-指向数组的指针变量例:编写程序,将数组中的数按颠倒的顺序重新存放。在操作时,只能借助一个临时存储单元而不得另外开辟数组。分析:不是要求按颠倒的顺序打印数据,而是要求按逆序重新放置数组中的内容。假定a数组中有8个元素第36页/共70页指针应用举例-指向数组的指针变量102030405060708010011003100510071009100B100D100Fa0a1a2a3a4a5a6a7807060504030201010011003100510071009100B
18、100D100Fa0a1a2a3a4a5a6a7第37页/共70页指针应用举例-指向数组的指针变量(1)定义两个变量i和j,首先将a0,a7进行对调,中间用到一个临时存储单元,所以定义同类型变量temp。(2)将变量i加1,j减1,满足条件ij时,将a1和a6进行对调(3)最后将a3和a4进行对调,此时i+1,j-1不再满足条件ij第38页/共70页指针应用举例-指向数组的指针变量#include stdio.h#define NUM 8void invert(int*,int);void priout(int*,int);main()int aNUM=10,20,30,40,50,60,70
19、,80;printf(nOutput primary data:);priout(a,NUM);invert(a,NUM);printf(nOutput the inverse data:);priout(a,NUM);第39页/共70页指针应用举例-指向数组的指针变量void priout(int s,int n)int i;for(i=0;in;i+)printf(%4d,si);printf(n);void invert(int*a,int n)int i,j,temp;for(i=0,j=n-1;ij;i+,j-)temp=*(a+i);*(a+i)=*(a+j);*(a+j)=tem
20、p;第40页/共70页指针应用举例-指向数组的指针变量例:w数组中存放了n个数据,编写函数删除下标为k的元素的值。第41页/共70页指针应用举例-指向数组的指针变量2122232425262728293010011003100510071009a0a1a2a3a4a5a6a7a8a9k262728293021222324262728293030*10011003100510071009a0a1a2a3a4a5a6a7a8a9第42页/共70页指针应用举例-指向数组的指针变量getindex:用于输入所删除元素的下标,函数中对输入的下标进行检查,若越界,则要求重新输入,直到正确为止。arrout
21、:用于输出数组中的元素arrdel:进行所要求的删除操作第43页/共70页指针应用举例-指向数组的指针变量#include stdio.h#define NUM 10int arrdel(int*,int,int);void arrout(int*,int);int getindex(int n);第44页/共70页指针应用举例-指向数组的指针变量main()int n,d,aNUM=21,22,23,24,25,26,27,28,29,30;n=NUM;printf(Output primary data:n);arrout(a,n);d=getindex(n);n=arrdel(a,n,d
22、);printf(nOutput the data after delete:n);arrout(a,n);第45页/共70页指针应用举例-指向数组的指针变量getindex(int n)int p;doprintf(nEnter the index 0=p%d:,n);scanf(%d,&p);while(pn-1);return p;void arrout(int w,int m)int k;for(k=0;km;k+)printf(%5d,wk);printf(n);第46页/共70页指针应用举例-指向数组的指针变量int arrdel(int*a,int n,int k)int i;f
23、or(i=k;in-1;i+)ai=ai+1;n-;return n;第47页/共70页指针应用举例-指针指向字符串例:编写函数strlength(*s),函数返回指针s所指字符串的长度。相当于库函数strlen的功能。第48页/共70页指针应用举例-指针指向字符串#include stdio.h#include string.hint strlength(char*s)int n=0;while(*(s+n)!=0)n+;return n;main()char str=ABCDEF;int len1,len2;len1=strlength();len2=strlength(str);prin
24、tf(len1=%d,len2=%d,len1,len2);第49页/共70页指针应用举例-指针指向字符串例:编写程序从若干字符串中找出最小的串进行输出。#include stdio.h#include string.h#define N 20#define M 81第50页/共70页指针应用举例-指针指向字符串int getstr(char pM)char tM;/*开辟一个临时的字符串存储空间*/int n=0;printf(Enter string.a empty string to end.n);gets(t);while(strcmp(t,)strcpy(pn,t);n+;gets(
25、t);return n;第51页/共70页指针应用举例-指针指向字符串char*findmin(char(*a)M,int n)char*q;int i;q=a0;/*用q指向字符串数组中最小串的地址*/for(i=0;i0)q=ai;return q;第52页/共70页指针应用举例-指针指向字符串main()char sNM,*sp;int n;n=getstr(s);sp=findmin(s,n);puts(sp);第53页/共70页指针应用举例-函数指针指向函数的指针变量的定义#include“”double fun(int a,int*p)main()double(*fp)(int,i
26、nt*),y;int n;fp=fun;y=(*fp)(56,&n)说明:(1)函数名代表该函数的入口地址(2)(*fp)(int,int*)说明fp是一个指向函数的指针,这个函数的基类型是double(3)fp=fun把fun函数的地址赋给指针变量fp(4)y=(*fp)(56,&n)实现对该函数的调用。等于于 y=fun(56,&n)第54页/共70页指针应用举例-函数指针例:函数名或指向函数的指针作为实参#include#include double tran(double(*)(double),double(*)(double),double);double tran(double(*
27、f1)(double),double(*f2)(double),double x)return(*f1)(x)/(*f2)(x);第55页/共70页指针应用举例-函数指针main()double y,v;v=60*3.1416/180.0;/*v=600*/y=tran(sin,cos,v);printf(“tan(60)=%10.6fn”,y);y=tran(cos,sin,v);printf(“cos(60)=%10.6fn”,y);第56页/共70页指针应用举例-动态存储分配例:short int*pi;float*pf;pi=(short*)malloc(2);pf=(float*)m
28、alloc(4);说明:(1)因为malloc是void型的,所以需要强制转换类型(2)强制转换类型括号中的不可少,否则就变成了普通类型。if(pi!=NULL)*p=6;63.8pipf第57页/共70页指针应用举例-动态存储分配注意:由动态存储分配得到的存储单元没有名字,只能靠指针来引用它。一旦指针改变指向,则原存储单元及所存数据都将无法引用。通过调用malloc函数所分配的单元动态存储单元中没有确定的初值。在动态申请存储空间时,若不能确定数据类型所占字节数,可通过sizeof运算符来求得。第58页/共70页指针应用举例-动态存储分配例:pi=(int*)malloc(sizeof(int
29、);pf=(float*)malloc(sizeof(float);由系统来计算指定类型所占的字节数,有利于程序的移植。第59页/共70页指针应用举例-链表单链表struct slistint data;struct slist *next;typedef struct slist SLIST;/0head第60页/共70页指针应用举例-链表(1)建立带有头结点的单向链表步骤:读取数据生成新结点将数据存入结点的成员变量中将新结点插入到链表中。(重复操作直到输入结束)。第61页/共70页指针应用举例-链表例:编写函数creat_slist,建立如上图所求的单向链表。结点数据域的内容从键盘输入,以
30、-1作为输入结束标志。链表头结点的地址由函数值返回。h:头指针,存放头结点的地址r:指向链表当前的尾结点s:指向新生成的结点每当s把新开辟的结点链接到链表尾后,r便移向这一新的表尾结点。第62页/共70页#include stdio.h#include stdlib.hstruct nodeint data;struct node*next;typedef struct node SLIST;main()SLIST*head;.head=return creat_slist();/*调用链表建立函数,得到头结点地址*/SLIST*creat_slist()SLIST*h,*s,*r;int c
31、;h=(SLIST*)malloc(sizeof(SLIST);/*生成头结点*/r=h;scanf(%d,&c);while(c!=-1)s=(SLIST*)malloc(sizeof(SLIST);/*生成新结点*/r-data=c;r-next=s;r=s;/*使r指向当前表尾*/scanf(%d,&c);r-next=0;/*置链表结束标志*/return h;0headh-next=0指针应用举例-链表第63页/共70页(1)顺序访问链表中各结点的数据域步骤:指向指向一个结点输出数据域的内容(重复上面两步)直到链表结束为止。数据域内容为0例:编写函数print_slist,顺序输出单
32、向链表各结点数据域中的内容。指针应用举例-链表第64页/共70页指针应用举例-链表void print_slist(SLIST*head)SLIST*p;p=head-next;/*p指向头结点后的第一个结点*/if(p=0)printf(Linklist is NULL!n);elseprintf(head);doprintf(-%d,p-data);/*输出当前结点数据域中的值*/p=p-next;while(p!=0);printf(-endn);第65页/共70页(2)在单向链表中插入结点s:用来指向新开辟的结点p:指向插入的位置q:p的前趋结点xyqps链表非空,值为x结点存在,插在
33、x 结点前链表非空,值为x结点不存在,插在表尾链表为空,插在表尾,即头结点之后。s-next=p;q-next=s;指针应用举例-链表第66页/共70页指针应用举例-链表insert_snode(SLIST*head,int x,int y)SLIST*s,*p,*q;s=(SLIST*)malloc(sizeof(SLIST);s-data=y;/*在新结点中存入y的值*/q=head;p=head-next;/*指针初始化,p指向第一个结点*/while(p!=0)&(p-data!=x)/*表非空,且未到表尾,查找x的位置*/q=p;p=p-next;/*q指向p的前趋结点*/s-nex
34、t=p;q-next=s;第67页/共70页(3)删除单向链表中的结点/87qp9头结点头指针q-next=p-next指针应用举例-链表第68页/共70页指针应用举例-链表struct link *delete_link(struct link *head,int x)struct link *q,*p;q=head;p=head-next;while(p!=0)&(p-data!=x)/*寻找被删除结点m*/q=p;/*q始终指向m之前的结点*/p=p-next;if(m=0)/*不存在符合条件的结点*/printf(“can not exit!n”);else q-next=p-next;/*逻辑删除结点*/free(m);/*物理删除结点*/return(head);第69页/共70页感谢您的观看!第70页/共70页