《C语言程序设计》课件ch6_第1页
《C语言程序设计》课件ch6_第2页
《C语言程序设计》课件ch6_第3页
《C语言程序设计》课件ch6_第4页
《C语言程序设计》课件ch6_第5页
已阅读5页,还剩72页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

第6章指针

指针是C语言中的一个重要概念,是其最具特色的语言部分,也是C语言的精华,同时也是C语言的难点。指针可以有效地表示复杂的数据结构;能动态分配内存;方便地使用字符串;有效而方便地使用数组;在调用函数时能获得1个以上的结果;能直接处理内存单元的地址等。6.1.1地址和指针的概念

内存是计算机的一个重要组成部分,用于存放那些“正在”使用的数据和“正在”执行的程序。内存单元的基本单位是字节,为了方便对内存的访问,每一个内存单元都有一个编号,就像门牌号一样,这个编号就是内存的地址。C程序中的每一个变量,C程序的每一个函数,在内存中都会对应一定的内存单元。

1.内存地址——内存中存储单元的编号6.1.1地址和指针的概念

编译时系统分配2000和2001两个字节给变量i,2002和2003两个字节给变量j,2004和2005两个字节给变量k。2.变量的地址——系统分配给变量的内存单元的起始地址

如果在程序中定义了一个变量,在对程序进行编译时,系统就会给这个变量分配一个或若干个存储单元。不同的数据类型分配不同字节的存储空间。

每一字节都有一个地址,一般把每个数据的首字节地址称为该数据的地址。假如有如下定义:inti,j,k;6.1.1地址和指针的概念3.变量值的存取——通过变量在内存中的地址进行在程序中一般是通过变量名来对内存单元进行存取操作的。其实程序经过编译以后己经将变量名转换为变量的地址,对变量值的存取都是通过地址进行的。6.1.1地址和指针的概念②系统对语句“scanf("%d",&i);”在执行时,如果从键盘输入3,表示要把3送到变量i中,实际上是把3送到地址为2000开始的整型存储单元中。(1)直接访问——直接利用变量的地址进行存取。①对于语句“printf("%d",i);”它是这样执行的:根据变量名与地址的对应关系(这个对应关系是在编译时确定的),找到变量i的地址2000,然后从由2000开始的两个字节中取出数据(即变量的值),把它输出。这里输出的值为3。

则从2000和2001字节取出i的值(3),再从2002和2003字节取出j的值(假设为6),将它们相加后再将其和(9)送到k所占用的2004和2005字节单元中。这种按变量地址存取变量值的方式称为“直接访问”方式。如果有语句:k=i+j;6.1.1地址和指针的概念

要存取变量i的值,也可以采用间接方式:先找到存放“i的地址”的变量i_pointer,从中取出i的地址(2000),然后到2000和2001字节取出i的值(3)。

(2)间接访问——通过另外一个变量访问该变量的值。C语言规定,可以在程序中定义整型变量、浮点型变量、字符变量等,也可以定义这样一种特殊的变量,它是存放地址的。

假设定义了一个变量i_pointer,用来存放整型变量的地址,它被分配为3010和3011两个字节。可以通过下面的语句将i的起始地址(2000)存放到i_pointer中。i_pointer=&i;这时,i_pointer的值就是2000,即变量i所占用单元的起始地址。6.1.1地址和指针的概念图6-2(a)表示直接访问,已经知道变量i的地址,根据此地址直接对变量i的存储单元进行存取访问(图示把数值3存放到i中)。6-2(a) 6-2(b)间接访问(3)两种访问方式的比较。图6-2(b)表示间接访问,先找到存放变量i地址的变量i_pointer,从其中得到变量i的地址,再据此找到变量i的存储单元,然后对它进行存取访问。6.1.1地址和指针的概念指针变量的值(即指针变量中存放的值)是地址(即指针)。一个变量的地址称为该变量的“指针”。例如,地址2000是变量i的指针。如果有一个变量专门用来存放另一个变量的地址(即指针),则它称为“指针变量”,指针变量就是地址变量(存放地址的变量)。6.1.2指向变量的指针变量定义了两个整型变量x和y,一个指针变量p。2.指针变量的定义与应用

与一般变量的定义相比,除变量名前多了一个星号“*”(指针变量的定义标识符)外,其余一样:数据类型*指针变量[,指针变量2……];例如:intx=10,*p,y;6.1.2指向变量的指针变量#include"stdio.h"voidmain(){intnum_int=12,*p_int; floatnum_f=3.14,*p_f; charnum_ch='p',*p_ch; p_int=&num_int; p_f=&num_f; p_ch=&num_ch; printf("num_int=%d,*p_int=%d\n",num_int,*p_int);printf("num_f=%d,*p_f=%d\n",num_f,*p_f);printf("num_ch=%d,*p_ch=%d\n",num_ch,*p_ch);}程序的运行结果为:num_int=12,*p_int=12num_f=3.14,*p_f=3.14num_ch=p,*p_ch=p【例6.1】指针变量的定义。/*定义一个指向int型数据的指针变量p_int*//*定义一个指向float型数据的指针变量p_f*//*定义一个指向char型数据的指针变量p_ch*//*取变量num_int的地址,赋值给p_int*//*取变量num_f的地址,赋值给p_f*//*取变量num_ch的地址,赋值给p_ch*/6.1.2指向变量的指针变量P=&x;y=*p;

指针变量p与整型变量x的关系

取地址运算的格式:&变量例如:intx=10,*p,y;/*取变量x的地址赋给指针变量p*//**p表示取指针变量p所指单元的内容,即变量x的值,则y=10*/6.1.2指向变量的指针变量if(p==NULL){……}其含义是:判断指针p是否为空,若指针p为空,则表达式成立。

在C语言中,如果intx=10,y,*px;且有px=&x;则变量x与指向变量x的指针px之间有以下等价关系:y=x;等价于y=*px;/*y=指针变量px的内容*/x++;等价于(*px)++;/*对指针变量px的内容加1*/y=x+5等价于y=(*px)+5;y=x;等价于y=*(&x);/*y=变量x地址的内容*/p=NULL;表示指针p为空,没有指向任何对象。C语言中用NULL表示空指针。6.1.2指向变量的指针变量#include"stdio.h"voidmain(){intnum1,num2;int*num1_p=&num1,*num2_p=&num2,*pointer;printf("inputthefirstnumber:");scanf("%d",num1_p);printf("inputthesecondnumber:");scanf("%d",num2_p);printf("num1=%d,num2=%d\n",num1,num2);fi(*unm1_p>*num2_p){pointer=numi_p;num1_p=num2_p;num2_p=pointer;}printf("min=%d,max=%d\n",*num1_p,*num2_p);}【例6.2】输入两个整数,按升序(从小到大排序)输出。/*如果num1>num2,则交换指针*/程序运行情况如下:inputthefirstnumber:9↙inputthesecondnumber:6↙num1=9,num2=6min=6,max=96.2指针与数组引用数组元素可以用下标法(如a[3]),也可以用指针的方法,即通过指向数组元素的指针找到所需的元素。

一个变量有一个地址,一个数组包含若干个元素,每个数组元素都在内存中占有存储单元,它们都有相应的地址。指针变量既然可以指向变量,也可以指向数组和数组元素。数组的指针是指数组的起始地址,数组元素的指针就是数组元素的地址。6.2.1指向数组元素的指针指向数组元素的指针变量的定义,与指向普通变量的指针变量的定义方法一样。例如:intarray[10];(定义array为包含10个整型变量的数组)int*pointer;(定义pointer为指向整型变量的指针变量)对指针变量的赋值:pointer=&array[0];

把array[0]元素的地址赋给指针变量pointer。也就是说,pointer指向array数组的第0号(第1个)元素。6.2.1指向数组元素的指针C语言规定数组名代表数组的首地址,也就是第一个元素的地址。

数组名不代表整个数组,上述“pointer=array;”的作用是“把array数组的第一个元素的地址赋给指针变量pointer”,而不是“把array数组各元素的值赋给pointer”。下面两个语句等价:pointer=&array[0];pointer=array;注意:6.2.1指向数组元素的指针

它的作用是把array数组的第一个元素的地址赋给指针变量pointer。在定义指针变量时可以对它赋初值。int*pointer=&array[0];它等效于:int*pointer;pointer=&array[0];定义时也可以写成:int*pointer=array;6.2.2通过指针引用数组元素

指向数组的指针变量,也可将其看作数组名,因而可按下标法来使用。1.通过指针引用一维数组中的元素如果有intarray[10],*pointer=array;则:*(pointer+i)和*(array+i)就是数组元素array[i]。pointer+i和array+i都是数组元素array[i]的地址。例如,pointer[i]等价于*(pointer+i)。6.2.2通过指针引用数组元素#include"stdio.h"#include"ctype.h"main(){intarray[10],*pointer=array,i;printf("Input10numbers:");for(i=0;i<10;i++)scanf("%d",pointer+i); printf("array[10]:");for(i=0;i<10;i++)printf("%d",*(pointer+i));printf("\n");}程序的运行情况如下:Input10numbers:0123456789↙Array[10]:0123456789【例6.3】使用指向数组的指针变量来引用数组元素。/*使用指针变量来输入数组元素的值*//*使用指向数组的指针变量输出数组*/6.2.2通过指针引用数组元素(4)关系运算:表示两个指针所指地址之间、位置的前后关系:前者为小,后者为大。说明:(1)指针变量的值是可以改变的,所以必须知道其当前值,否则容易出错。(2)指向数组的指针变量,可以指向数组以后的内存单元,但是没有实际意义。(3)对指向数组的指针变量(px和py)进行算术运算和关系运算的含义如下:px±n:将指针当前位置向前(+n)或后退(-n)n个数据单位,而不是n个字节px++/++px,px--/--px是px±n的特例(n=1)。px-py:两指针之间的数据个数,而不是指针的地址之差。6.2.2通过指针引用数组元素其中:a为数组的首地址,M和N分别为二维数组行和列的元素个数。2.通过指针引用二维数组中的元素

在C语言中,二维数组是按行优先的规律转换为一维线性数组存放在内存中的,因此,可以通过指针访问二维数组中的元素。如果有:inta[M][N];则将二维数组中的元素a[i][j]转换为一维线性地址的一般公式是:线性地址=a+i×M+j6.2.2通过指针引用数组元素则二维数组a的数据元素在内存中的存储顺序及地址关系如图6-4所示。若有:inta[4][3],*p;P=&a[0][0];图6-4二维数组的元素在内存中的存储顺序及地址关系数组名称含义a[0]a[0][0]一维下标的指针二维数组的下标表示元素在内存中的存储顺序通过指针访问元素通过指针按下标访问元素aa[1]a[2]a[3]a[0][1]a[0][2]a[1][0]a[1][1]a[1][2]a[2][0]a[2][1]a[2][2]a[3][0]a[3][1]a[3][2]pP+1P+2P+3P+4P+5P+6P+7P+8P+9P+10P+11p[0]p[1]p[2]P[3]p[4]p[5]p[6]p[7]p[8]p[9]p[10]p[11]6.2.2通过指针引用数组元素数组元素a[i][j]的存储地址是:&a[0][0]+i*n+j。a和a[0]是数组元素a[0][0]的地址,也是第1行的起始地址。a+1和a[1]是数组元素a[1][0]的地址,也是第2行的起始地址。由于a是二维数组,经过两次下标运算[]之后才能访问到数组元素。所以根据C语言的地址计算方法,a要经过两次*操作后才能访问到数组元素。6.2.2通过指针引用数组元素算法分析:给定的月是i,则将1,2,3,…,i-1月的各月的天数累加,再加上指定的日。但对于闰年,二月的天数为29天,因此还要判断给定的年是否为闰年,为了实现这一算法,需要设置一张每月天数的列表,给出每月的天数,考虑闰年非闰年的情况,此表可以设置成一个两行13列的二维数组,其中第1行对应的每列(设1~12列有效)元素是平年各月的天数,第2行对应的是闰年各月的天数。【例6.4】给定某年某月某日,将其转换成这一年的第几天并输出。6.2.2通过指针引用数组元素#include"stdio.h"#include"ctype.h"main(){Staticintday_tab[2][13]={{0,31,28,31,30,31,30,31,31,30,31,30,31},{0,31,29,31,30,31,30,31,31,30,31,30,31}};inty,m,d;scanf("%d%d%d",&y,&m,&d);printf("%d\n",day_of_year(day_tab,y,m,d)); }day_of_year(day_tab,year,month,day)int*day_tab; intyear,month,day;{inti,j;i=(year%4==0&&year%100!=0)||year%400==0;for(j=1;j<month;j++)day+=*(day_tab+i*13+j);return(day);}/*实参为二维数组名*//*形式参数为指针*//*day_tab+i*13+j;对二维数组中元素进行地址变换*/6.2.3数组名作为函数参数当数组名作为实参时,如果形参数组中各元素的值发生了变化,实参数组元素的值也随之变化。数组名作为函数的形参和实参。如:main(){intarray[10];f(array,10);}f(arr,n)intarr[],n;{

…}array为实参数组名,arr为形参数组名6.2.3数组名作为函数参数

在函数调用前,形参数组并不占用内存,在函数调用的时候,形参数组并没有另外开辟新的存储单元,而是以实参数组的首地址作为形参数组的首地址,这样实参数组array的第1个元素和形参数组arr的第1个元素共占一个内存单元。同理,array[1]与arr[1]共占一个内存单元。如果在函数调用过程中使形参数组arr的元素值发生变化也就使得实参数组的元素值发生了变化。实参数组和形参数组各元素之间并不存在“值传递”。6.2.3数组名作为函数参数算法分析:将a[0]与a[n-1]对换,再将a[1]与a[n-2]对换,……,直到将a[(n-1)/2]与a[n-int((n-1)/2)]对换。现在用循环处理这个问题,设两个“位置指示变量”i和j,i的初始值为0,j的初始值为n-1。将a[i]和a[j]交换,然后使i的值加1,j的值减1,再将a[i]和a[j]对换,直到i=(n-1)/2为止。【例6.5】将数组a中n个整数按照相反顺序存放。6.2.3数组名作为函数参数voidmain(){inti,a[10]={3,1,9,11,0,6,7,5,4,2};for(i=0;i<10;i++)printf("%d,",a[1]);printf("\n");inv(a,10);for(i=0;i<10;i++)printf("%d,",a[i]);printf("\n");}运行情况如下:3,1,9,11,0,6,7,5,4,22,4,5,7,6,0,11,9,1,3#include"stdio.h"voidinv(intx[],intn){inttemp,i,j,m=(n-1)/2;for(i=0;i<m;i++){j=n-1-i;temp=x[i];x[i]=x[j];x[j]=temp;}return;}/*形参x是数组名*/6.2.3数组名作为函数参数

对这个程序可以做一些改动。将函数inv中的形参x改为指针变量。实参为数组名a,即数组a的首地址,传递给形参指针变量x,x就指向a[0]。x+m是a[m]元素的地址。设i和j以及p都是指针变量,用它们指向有关元素。i的初始值为x,j的初始值为x+n-1,见图6-6。使*i和*j交换就是使a[i]与a[j]交换。图6-6指针变量与数组的关系6.2.3数组名作为函数参数voidmain(){inti,a[10]={3,7,9,11,0,6,7,5,4,2};for(i=0;i<10;i++)printf("%d,",a[1]);printf("\n");inv(a,10);for(i=0;i<10;i++)printf("%d,",a[i]);printf("\n");}【例6.6】程序如下:#include"stdio.h"voidinv(int*x,intn){intp,temp,*i,*j,m=(n-1)/2;i=x;j=x+n-1;p=x+m;for(;i<=p;i++,j--){temp=*x;*x=*j;j=temp;}return;}6.2.3数组名作为函数参数main(){inti,number[10];for(i=0;i<10;i++)scanf("%d",&number[i]);max_min_value(nember,10);printf("\nmax=%d,min=%d\n",max,min);}运行结果如下:-567675489034-2319↙max=89,min=-23【例6.7】从10个数中找出其中最大值和最小值。算法分析:本题要求不改变数组元素的值,只须找出该数组中的最大值与最小值,而函数只能得到一个返回值,现在用全局变量在函数之间传递数据。#include"stdio.h"intmax,min;voidmax_min_value(intarray[],intn){intp,array_end;array_end=array+n;max=min=*array;for(p=array+1;p<array_end;p++)if(*p>max)max=*p;elseif(*p<min)min=*p;return;}/*全局变量*/6.2.3数组名作为函数参数程序可改为:#include"stdio.h"intmax,min;voidmax_min_value(int*array,intn){int*p,*array_end;array_end=array+n;max=min=*array;for(p=array+1;p<array_end;p++)if(*p>max)max=*p;elseif(*p<min)min=*p;return;}实参也可以不用数组名,而用指针变量传递地址。main(){inti,number[10],*p;p=number;for(i=0;i<10;i++)scanf("%d",p);for(p=number,i=0;i<10;i++,p++)printf("%d",*p);p=number;max_min_value(p,10);printf("\nmax=%d,min=%d\n",max,min);}6.2.3数组名作为函数参数形参用数组名、实参用指针变量。

引入指向数组的指针变量后,数组及指向数组的指针变量作函数参数时,可有4种等价形式(本质上是一种,即指针数据作函数参数):形参和实参都用数组名。形参和实参都用指针变量。形参用指针变量、实参用数组名。6.2.4指针与字符数组

在C语言中,既可以用字符数组表示字符串,也可以用字符指针变量来表示;引用时,既可以逐个字符引用,也可以整体引用。

字符串在内存中的起始地址称为字符串的指针,可以定义一个字符指针变量指向一个字符串。1.字符串的表示和引用6.2.4指针与字符数组#include"stdio.h"main(){char*string="IloveBeijing.";for(;*string!='\0';string++)printf("%c",*string);printf("\n");}也可以分成如下两条语句:char*string;string="IloveBeijing.";程序运行结果:IloveBeijing(1)逐个引用。【例6.9】使用字符指针变量表示和引用字符串。6.2.4指针与字符数组#include"stdio.h"main(){char*string="IloveBeijing.";printf("%s",*string);}(2)整体引用。【例6.10】采用整体引用的办法改写例6.9。注意:其他类型的数组是不能用数组名来一次性输出它的全部元素的,只能逐个元素输出。对字符串中字符的存取可以用下标方法,也可以用指针方法。6.2.4指针与字符数组#include"stdio.h"main(){chara[]="HelloWorld!",b[20];inti;for(i=0;*(a+1)!='\0';i++)*(b+i)=*(a+i);*(b+i)='\0';printf("stringais:%s\n",a);printf("stringbis:");for(i=0;b[i]!='\0';i++)printf("%c",b[i]);printf("\n");}程序运行结果为:stringais:HelloWorld!stringbis:HelloWorld!【例6.11】将字符串a复制到字符串b中。6.2.4指针与字符数组#include"stdio.h"main(){chara[]="HelloWorld!",b[20],*p1,*p2;inti;p1=a;p2=b;for(;*p1!='\0';p1++,p2++)*p2=*p1;*p2='\0';printf("stringais:%s\n",a);printf("stringbis:");for(i=0;b[i]!='\0';i++)printf("%c",b[i]);printf("\n");}也可以设指针变量,用其值的改变来指向字符串中不同的字符。【例6.12】用指针变量来处理例6.11的问题。6.2.4指针与字符数组2.字符指针变量与字符数组的比较(3)指针变量的值是可以改变的,字符指针变量也不例外。(1)存储内容不同。(2)赋值的方式不同。字符数组中存储的是字符串本身。字符指针变量中存储的是字符串的首地址;字符指针变量赋值语句:char*pointer;pointer="Thisisaexample.";charchar_array[20];char_array="Thisisaexample.";/*非法的用法*/字符数组虽然可以在定义时初始化,但不能用赋值语句整体赋值。6.2.4指针与字符数组charstr[]={"IloveChina!"};str=str+7;printf("%s",str);main(){char*a="IloveChina!";a=a+7;printf("%s",a);}【例6.13】利用指针字符变量输出一串字符。程序运行的结果为:China!下面是错误的:6.2.4指针与字符数组#include"stdio.h"main(){char*a="ILOVECHINA!";inti;printf("Thesixthcharacteris%c\n",a[5]);for(i=0;a[i]!='\0';i++)printf("%c",a[i]);}【例6.14】程序运行的结果为:ThesixthcharacterisEILOVECHINA!6.2.4指针与字符数组3.字符串指针作为函数参数#include"stdio.h"voidstring_copy(char*str_form,char*str_to){inti=0;for(;(*(str_to+i)=*(str_form+i))!='\0';i++); }main(){chararray_str1[20]="Iamateacher.";chararray_str2[20];string_copy(array_str1,array_str2); printf("array_str2=%s\n",array_str2);}【例6.15】用函数调用方式,实现字符串的复制。程序的运行结果:Iamateacher./*循环体为空语句*//*数组名作为实参*/6.2.4指针与字符数组其特点是:先复制、后判断,循环结束前,结束标志已经复制。

“for(;(*(str_to+i)=*(str_form+i))!='\0\;i++);”语句的执行过程为:首先将源串中的当前字符复制到目标串中;然后判断该字符(即赋值表达式的值)是否为结束标志。如果不是,则相对位置变量增1,以便复制下一个字符;如果是结束标志,则结束循环。6.2.5数组指针

它表明指针p指向的存储空间有4个整型元素,即数组指针p指向一个一维数组,p的值就是该一维数组的首地址。

如果定义一个指针p,让它指向一个包含n个元素的一维数组,且p的增值以一维数组的长度为单位,此时,如果指针p指向二维数组的某一行,则p+1就指向了该二维数组的下一行。在C++中,这样的指针称为数组指针,使用数组指针可以很方便地处理二维数组。数组指针的说明形式如下:存储类型数据类型(*指针名)[元素个数]例如:在程序中定义一个数组指针:int(*p)[4];6.2.5数组指针(2)p是一个行指针,它只能指向一个包含n个元素的一维数组,不能指向一维数组中的元素,如果要访问一维数组中的某个元素,如第j个元素,可用(*p)[j]表示。在使用数组指针时,有两点一定要注意:(1)*p两侧的括号一定不能漏掉,如果写成*p[4]的形式,由于[]的运算级别高,因此p先和[4]结合,是数组,然后再与前面的*结合,*p[4]是指针数组。6.2.5数组指针#include"stdio.h"voidmain(){inta[2][3]={1,3,5,7,9,11};int(*p)[3];inti,j;for(i=0;i<2;i++){for(j=0;j<3;j++)printf("%2d",(*p)[i]);p++;}}【例6.16】用数组指针处理二维数组。程序的运行结果为:13579116.3指针与函数

函数之间可以传递一般变量的值,也可以传递地址(指针)。函数与指针之间有着密切的关系,它包含3种含义:指针作为函数的参数,函数的返回值为指针以及指向函数的指针。6.3.1指针作函数参数指针变量,既可以做函数的形参,也可以作函数的实参。指针变量作实参时,与普通变量一样,也是“值传递”,即将指针变量的值(一个地址)传递给被调用函数的形参(必须是一个指针变量)。6.3.1指针作函数参数#include"stdio.h"voidexchange(int*pointer1,int*pointer2){inttemp;temp=*pointer1;*pointer1=*pointer2;*pointer2=temp;}voidmain(){intnum1,num2;int*num1_p=&num1,*num2_p=&num2;printf("Inputthefirstnumber:");scanf("%d",num1_p);printf("Inputthesecondnumber:");scanf("%d",num2_p);printf("num1=%d,num2=%d\n",num1,num2);if(*num1_p>*num2_p)exchange(num1_p,num2_p);printf("min=%d,max=%d\n",num1,num2);}程序的运行情况如下:Inputthefirstnumber:9↙Inputthesecondnumber:6↙num1=9,num2=6min=6,max=9【例6.17】使用函数调用方式改写例6.2,要求实参为指针变量。/*定义并初始化指针变量num1_p和num2_p*//*即num1>num2*//*指针变量作实参*//*输出排序后的num1和num2的值*/6.3.1指针作函数参数

其机制为:在执行被调用函数时,使形参指针变量所指向的变量的值发生变化;函数调用结束后,通过不变的实参指针(或实参指针变量)将变化的值保留下来。虽然被调用函数不能改变实参指针变量的值,但可以改变它们所指向的变量的值。为了利用被调用函数改变的变量值,应该使用指针(或指针变量)作函数实参。6.3.1指针作函数参数/*排序*/if(num1<num2)exchange(&num1,&num2);if(num1<num3)exchange(&num1,&num3);if(num2<num3)exchange(&num2,&num3);/*输出排序结果*/printf("排序结果:%d,%d,%d\n",num1,num2,num3);}【例6.18】输入3个整数,按降序(从大到小的顺序)输出。要求使用变量的指针作函数调用的实参来实现。#include"stdio.h"voidexchange(int*pointer1,int*pointer2){inttemp;temp=*pointer1;*pointer1=*pointer2;*pointer2=temp;}voidmain(){intnum1,num2,num3;/*从键盘上输入3个整数*/printf("Inputthefirstnumber:");scanf("%d",num1);printf("Inputthesecondnumber:");scanf("%d",num2);printf("Inputthethirdnumber:");scanf("%d",num3);printf("num1=%d,num2=%d,num3=%d",num1,num2,num3)程序的运行情况如下:Inputthefirstnumber:9↙Inputthesecondnumber:6↙Inputthethirdnumber:12↙num1=9,num2=6,num3=12排序结果:12,9,66.3.2函数指针1.函数指针的概念

一个函数在编译时被分配了一个入口地址,这个地址就称为该函数的指针。可以用一个指针变量指向一个函数,然后通过该指针变量调用此函数。6.3.2函数指针#include"stdio.h"voidmain(){intmax();int(*p)();inta,b,c;p=max;scanf("%d,%d",&a,&b);c=(*p)(a,b);printf("a=%d,b=%d,max=%d",a,b,c);}intmax(intx,inty){intz;if(x>y)z=x;elsez=y;return(z);}【例6.19】求a和b中的大者。6.3.2函数指针注意:对指向函数的指针变量,诸如p+i、p++∕p--等运算是没有意义的。2.指向函数的指针变量(1)定义格式:函数类型(*指针变量)();注意:“(*指针变量)”外的括号不能缺,否则就成了返回指针值的函数。例如:int(*fp)();/*fp为指向int函数的指针变量*∕(2)赋值。函数名代表该函数的入口地址。

可用函数名给指向函数的指针变量赋值。指向函数的指针变量=[&]函数名;注意:函数名后不能带括号和参数;函数名前的“&”符号是可选的。(3)调用格式:(*函数指针变量)([实参表])

函数名作实参时,因为要缺省括号和参数,造成编译器无法判断它是一个变量还是一个函数,所以必须加以说明。6.3.2函数指针

指向函数的指针变量的常用用途之一就是将函数指针作参数传递到其他函数。3.指向函数的指针变量作函数参数6.3.2函数指针【例6.20】设一个函数process,在调用它时,每次实现不同的功能。输入a和b两个数,第一次调用process时找出a和b中大者,第二次找出其中小者,第三次求a和b之和。#include"stdio.h"main(){intmax(int,int); intmin(int,int); intadd(int,int); inta,b;scanf("%d,%d",&a,&b);printf("max=");processs(a,b,max);printf("min=");process(a,b,min);printf("sum=");process(a,b,add);}/*函数声明*//*函数声明*//*函数声明*/6.3.2函数指针max(intx,inty) {intz;if(x>y)z=x;elsez=y;return(z);}min(intx,inty) {intz;if(x<y)z=x;elsez=y;return(z);}add(intx,inty) {intz;z=x+y;return(z);}process(intx,inty,int(*fun)(int,int)){intresult;result=(*fun)(x,y);printf("%d\n",result);}程序运行结果如下:3,9↙max=9min=3sum=12/*表示fun是指向函数的指针,该函数是一个整型函数,有两个整型形参*//*函数定义*//*函数定义*//*函数定义*/6.3.3指针函数pr是函数名,调用它以后能得到一个指向整型数据的指针(地址)。x、y是函数pr的形参。

在*pr两侧没有括号,在pr的两侧分别为*运算符和()运算符。而()的优先级高于*,因此pr先与()结合。显然这是函数形式。这个函数前面有一个*,表示此函数是指针型函数(函数值是指针)。最前面的int表示返回的指针指向整型变量。

指针函数是指返回值为指针的函数,函数的返回值可以是不同类型的指针。指针函数定义格式:函数类型*函数名([形参表])例如:int*pr(x,y);注意:6.3.3指针函数int*seek(int(*pnt_row)[3]){inti=0,*pnt_col; pnt_col=*(pnt_row+1); for(;i<3;i++)if(*(*pnt_row+i)<60){pnt_col=*pnt_row; break;}return(pnt_col);}【例6.21】某数理化三项竞赛训练组有3个人,找出其中至少有一项成绩不合格者。要求使用指针函数实现。/*定义一个(列)指针变量pnt_col*//*使pnt_col指向下一行之首(作标志用)*//*某项成绩不合格*//*使pnt_col指向本行之首*//*退出循环*/6.3.3指针函数程序运行结果:No.1gradelist:556575main(){intgrade[3][3]={{55,65,75},{65,75,85},{75,80,90}};inti,j,*pointer; for(i=0;i<3;i++) {pointer=seek(grade+i); if(pointer==*(grade+i)) {printf("No.℅dgradelist:",i+1);for(j=0;j<3;j++)printf("℅d",*(pointer+j));printf("\n");}}}

/*定义一个(列)指针变量pointer*//*控制每个学生*//*用行指针作实参,调用seek()函数*//*该学生至少有一项成绩不合格*//*输出该学生的序号和各项成绩*/6.3.3指针函数if(*(*pnt_row+i)<60)行中的pnt_row是一个行指针,指向数组grade的第i行;*pnt_row使指针由行转换为列,指向数组grade的第i行第0列;*pnt_row+j的值还是一个指针,指向数组的第i行第j列;*(*pnt_row+j)是一个数据(数组元素grade[i][j]的值)。程序说明如下:(1)主函数中的pointer=seek(grade+i);语句调用seek()函数时,将实参grade+i(行指针)的值复制到形参pnt_row(行指针变量)中,使形参pnt_row指向grade数组的第i行。pnt_col=*(pnt_row+1);语句中的*(pnt_row+1)将行指针转换为列指针,指向grade数组的第i+1行第0列,并赋值给(列)指针变量pnt_col。(2)在指针函数seek()中:6.4.1多级指针

使指针p指向a,则指针p所指向的变量*p就是要处理的数据变量a。如果同时存在另一个指针pp,并且把指针p的地址赋予指针变量pp,即:pp=&p;,则pp就指向指针p,这时指针pp所指向的变量*pp就是指针p,pp就是一个二级指针。数据类型**指针变量[,**指针变量2,…];1.概念

一个指针变量可以指向一个相应数据类型的数据,例如:inta,*p;p=&a;2.定义格式6.4.1多级指针#include"stdio.h"main(){staticchar*name[]={"Followme","BASIC","FORTRAN","GreatWall","Computerdesign"};char**p;inti;for(i=0;i<5;i++){p=name+i;printf("%s\n",*p);}}程序运行结果如下:FollowmeBASICFORTRANGreatWallComputerdesign【例6.22】用二级指针处理多个字符串。6.4.1多级指针

此时为了得到数据“5”,可以先使p=num+2,然后输出**p。注意*p是p间接指向的对象地址。而**p是p间接指向的对象的值。指针数组的元素也可以不指向字符串,而指向整型数据或浮点型数据等。例如:inta[5]={1,3,5,7,9};int*num[5]={&a[0],&a[1],&a[2],&a[3],&a[4]};int**p;6.4.2指针数组

表示定义一个由5个指针变量构成的指针数组,数组中的每个数组元素——指针,都指向一个整数。1.概念

数组的每个元素都是一个指针数据。指针数组比较适合用于指向多个字符串,使字符串处理更加方便、灵活。2.定义格式数据类型*数组名[元素个数]例如:int*pa[5];6.4.2指针数组int(*pb)[5]表示定义了一个指向数组的指针pb,pb指向的数组是一维的,大小为5的整型数组,其结构如下所示。int*pa[5]与int(*pb)[5]的区别:int*pa[5]int(*pb)[5]6.4.2指针数组#include"stdio.h"voidsort(char*name[],intcount){char*temp_p;inti,j,min;/*使用选择法排序*/for(i=0;i<count-1;i++) /*外循环:控制选择次数*/{min=i; /*预置本次最小串的位置*/for(j=i+1;j<count;j++) /*内循环:选出本次的最小串*/if(strcmp(name[min],name[j])>0) /*存在更小的串*/min=j; /*保存之*/if(min!=i) /*存在更小的串,交换位置*/{temp_p=name[i];name[min];name[min]=temp_p;}}}【例6.23】有若干计算机图书,请按字母顺序从小到大输出书名。使用排序函数完成排序,在主函数中进行输入输出。6.4.2指针数组/*主函数main()*/main(){char*name[5]={"BASIC","FORTRAN","PASCAL","C","FoxBASE"};inti=0;

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论