C语言程序设计教程 第6章指针

上传人:小**** 文档编号:240743443 上传时间:2024-05-04 格式:PPT 页数:79 大小:594KB
返回 下载 相关 举报
C语言程序设计教程 第6章指针_第1页
第1页 / 共79页
C语言程序设计教程 第6章指针_第2页
第2页 / 共79页
C语言程序设计教程 第6章指针_第3页
第3页 / 共79页
点击查看更多>>
资源描述
7 7.1.1 指针与指针变量指针与指针变量7 7.2.2 指针与函数指针与函数7 7.3.3 指针与数组指针与数组7 7.4.4 指针与字符串指针与字符串7 7.5.5 指针数组与命令行参数指针数组与命令行参数7 7.6.6 程序举例程序举例第 7 章 指针C C语言程序设计教程语言程序设计教程5/4/20241C语言程序设计教程第8章 指针7.1 7.1 指针与指针变量指针与指针变量7.1.1 7.1.1 7.1.1 7.1.1 指针的概念指针的概念指针的概念指针的概念1.1.内存与变量地址内存与变量地址内存地址:内存地址:内存是计算机用于存储数据的存储器,以一个字节作为存储单元,为了便于访问,给每个字节单元一个唯一的编号,第一字节单元编号为0,以后各单元按顺序连续编号,这些单元编号称为内存单元的地址。变量地址:变量地址:变量所分配存储空间的首字节单元地址(字节单元编号)。5/4/20242C语言程序设计教程第8章 指针2.变量的三要素:名字、类型与值每个变量都通过变量名与相应的存储单元相连系,具体分配哪些单元给变量,由编译系统完成变量名到对应内存单元地址的变换。变量分配存储空间的大小由类型决定。变量的值则是指相应存储单元的内容。3.3.内存存取方式直接存取:把直接按变量名或地址存取变量值的方式称为“直接存取”方式。间接存取:通过定义一种特殊的变量专门存放内存或变量的地址,然后根据该地址值再去访问相应的存储单元。5/4/20243C语言程序设计教程第8章 指针系统为特殊变量p(用来存放地址的)分配的存储空间地址是4800,p中保存的是变量a的地址,即4000,当要读取a变量的值12345时,不是直接通过a变量,也不是直接通过保存12345的内存单元的地址4000去取值,而是先通过变量p得到p的值4000,即a的地址,再根据地址4000读取它所指向单元的值12345。这种间接的通过变量p得到变量a的地址,再存取变量a的值的方式即为“间接存取”。通常称变量p指向变量a,变量a是变量p所指向的对象400012345400040024800pap(4800)p(4800)a(4000)4000123454000间接存取示意图5/4/20244C语言程序设计教程第8章 指针4.指针的概念在语言中,用指针来表示一个变量指向另一个变量这样的指向关系。指针指针:一个变量的地址称为该变量的指针。指针变量:一个变量专门用来存放另一变量的地址(即指针),则这个变量称为指针变量.5/4/20245C语言程序设计教程第8章 指针7.1.2指针变量的定义与初始化指针变量的定义与初始化1.1.指针变量的定义指针变量的定义类型标识符类型标识符*指针变量名;指针变量名;例:float*p1;(定义p1为指向实型变量的指针变量)char*p2;(定义p2为指向字符型变量的指针变量)在指针变量定义中,*是一个说明符,它表明其后的变量是指针变量,如p是指针变量,而不要认为*p是指针变量。指针变量定义时指定的数据类型是指针变量所指向的对象的数据类型指针变量存放的是所指向的某个变量的地址值,而普通变量保存的是该变量本身的值指针变量并不固定指向一个变量,可指向同类型的不同变量5/4/20246C语言程序设计教程第8章 指针(1)指针运算符与地址运算符指针运算符与地址运算符与指针引用有关的两个运算符:&与*。&:取地址运算符 *:指针运算符,或称指向运算符、间接访问运算 符。指针指向的对象的表示形式:*指针变量此处*是访问指针所指对象的运算符,与指针定义时的*不同。2.指针变量初始化5/4/20247C语言程序设计教程第8章 指针(2)指针变量初始化若有定义:inta,*p;语句仅仅定义了指针变量p,但指针变量并未指向确定的变量(或内存单元)。因为这些指针变量还没有赋给确定的地址值,只有将某一具体变量的地址赋给指针变量之后,指针变量才指向确定的变量(内存单元)。指指针针变变量量初初始始化化:在定义指针时同时给指针一个初始值如:inta,*p=&a;&a3*papaa5/4/20248C语言程序设计教程第8章 指针(3)指针变量的引用*指针变量名指针变量名代表所指变量的值。代表所指变量的值。指针变量名指针变量名代表所指变量的地址。代表所指变量的地址。有定义:inta,*p=&a;用*p来表示p指向的对象a,*p与a是等价的。*p可以象普通变量一样使用。例如:a=12;*p=12;scanf(%d,&*p);scanf(%d,p);printf(“%d%d”,*p,a);注注意意:*与&具有相同的优先级,结合方向从右到左。这样,&*p即&(*p),是对变量*p取地址,它与&a等价;p与&(*p)等价,a与*(&a)等价。5/4/20249C语言程序设计教程第8章 指针7.1.3指针运算1.1.指针的赋值运算指针的赋值运算(1)将变量地址值赋给指针变量,使指针指向该变量。设有如下定义:inta,b,*pa,*pb;float*pf;第一行定义了整型变量a,b及指针变量pa,pb。pa、pb还没有被赋值,因此pa、pb没有指向任何变量,下面语句完成对pa,pb的赋值:pa=&a;pb=&b;5/4/202410C语言程序设计教程第8章 指针例如:intj,k;int*pointer1,*pointer2;pointer1=&j;pointer2=&k;pointer1jpointer2k5/4/202411C语言程序设计教程第8章 指针(2)相同类型的指针变量间的赋值相同类型的指针变量间的赋值pa与pb都是整型指针变量,它们间可以相互赋值,如:pb=pa;即pa,pb都指向变量a,此时a、*pa、*pb是等价的。指针指向变化如下图:注注意意:只有相同类型的指针变量才能相互赋值,如pf=pa;是不允许的。因为pa是整型指针,pf是浮点型指针。&a&bpapba,*pab,*pb&a&apapba,*pa,*pbb5/4/202412C语言程序设计教程第8章 指针()()给指针变量赋给指针变量赋空值空值给指针变量赋空值,说明该指针不指向任何变量。“空”指针值用NULL表示,NULL是在头文件stdio.h中预定义的常量,其值为,在使用时应加上预定义行,如:#includestdio.hint*pa=NULL;亦可以用下面的语句给指针赋“空值”:a=0;或:pa=0;这里指针pa并非指向0地址单元,而是具有一个确定的“空值”,表示pa不指向任何变量。注注意意:指针虽然可以赋值0,但却不能把其它的常量地址赋给指针。例如:pa=4000;是非法的。5/4/202413C语言程序设计教程第8章 指针例7.1指针定义与初始化main()inta,b;int*pointer_1,*pointer_2;a=100;b=10;pointer_1=&a;pointer_2=&b;printf(%d,%dn,a,b);printf(%d,%dn,*pointer_1,*pointer_2);5/4/202414C语言程序设计教程第8章 指针程序运行结果:100,10100,10&a&bbPointer_1Pointer_2a*pointer_1*pointer_25/4/202415C语言程序设计教程第8章 指针例7.2 从键盘上输入两个整数到a、b,按由大到小输出。#includemain()inta,b,*pa=&a,*pb=&b,*p;/*定义指针变量pa、pb,如下页图a*/scanf(%d%d,&a,&b);if(*pa*pb)p=pa;/*进行指针交换,如下页图b,c*/pa=pb;pb=p;printf(na=%d,b=%dn,a,b);printf(nmax=%d,min=%d,*pa,*pb);/*pa指向大数,pb指向小数*/5/4/202416C语言程序设计教程第8章 指针若输入:1222输出结果:a=12,b=22max=22,min=12(a)(b)(c)指针变化示意图1222ppapb1222ppapbpapbab5/4/202417C语言程序设计教程第8章 指针2.2.指针的算术运算指针的算术运算 (1)加减运算:一个指针可以加、减一个整数n,其结果与指针所指对象的数据类型有关。指针变量的值应增加或减少“nsizeof(指针类型)”。加加减减运运算算常常用用于于数数组组的的处处理理。对指向一般数据的指针,加减运算无实际意义。例如:int a10,*p=a,*x;x=p+3;/*实际上是p加上3*2个字节赋给x,x指向数组的第三个分量*/对于不同基类型的指针,指针变量“加上”或“减去”一个整数n所移动的字节数是不同的。例如:float a10,*p=a,*x;x=p+3;/*实际上是p加上3*4个字节赋给x,x依然指向数组的第三个分量*/5/4/202418C语言程序设计教程第8章 指针(2)自增自减运算指针变量自增、自减运算具有上述运算的特点,但有前置后置、先用后用的考虑,务请小心。例如:int a10,*p=a,*x;x=p+;/*x第一个元素分量,p指向第二个元素*/x=+p;/*x、p均指向数组的第二个分量*/*p+相当于*(p+)。*(p+)与(*p)+含义不同,前者表示地地址址自自增增,后者表示当前当前所指向的数据自增所指向的数据自增。5/4/202419C语言程序设计教程第8章 指针2.*&a的含意是什么?(答:a)3.(*p)+相当于什么?(答:a+)思考:1.若有定义inta,*p;执行了“p=&a”,则:“&*p”的含意是什么?(答:相当于&a)5/4/202420C语言程序设计教程第8章 指针 3.3.指针的关系运算指针的关系运算 与基本类型变量一样,指针可以进行关系运算。与基本类型变量一样,指针可以进行关系运算。在在关关系系表表达达式式中中允允许许对对两两个个指指针针进进行行所所有有的的关关系系运运算算。若若p,q是是两两个个同同类类型型的的指指针针变变量量,则则:pq,p=q都都是是允许的。允许的。指指针针的的关关系系运运算算在在指指向向数数组组的的指指针针中中广广泛泛的的运运用用,假假设设p、q是是指指向向同同一一数数组组的的两两个个指指针针,执执行行pq的的运运算算,其其含含义义为为,若若表表达达式式结结果果为为真真(非非值值),则则说说明明p所所指指元元素素在在q所所指指元元素素之之后。或者说后。或者说q所指元素离数组第一个元素更近些。所指元素离数组第一个元素更近些。注注意意:在在指指针针进进行行关关系系运运算算之之前前,指指针针必必须须指指向向确确定定的的变变量量或或存存储储区区域域,即即指指针针有有初初始始值值;另另外外,只只有有相相同同类类型型的的指指针针才才能进行比较。能进行比较。5/4/202421C语言程序设计教程第8章 指针7.1.4 7.1.4 多级多级指针指针 把指向指针型数据的指针变量称为指向指针的指针,或称称称称多多多多级级级级指针指针指针指针。二级指针的定义形式如下:数据类型 *指针变量例如:inta,*p,*pp;a=22;p=&a;pp=&p;假设变量a的地址为4000,指针p的地址为4100,二级指针pp的地址为4800。a、p、pp三者的关系如上图。&p=4100&a=400022pp(4800)p(4100)a(4000)5/4/202422C语言程序设计教程第8章 指针7.2 指针与函数7.2.1 7.2.1 指针作为函数参数指针作为函数参数利利用用指指针针作作函函数数参参数数,可可以以实实现现函函数数之之间间多多个个数数据据的的传传递递,当当形形参参为为指指针针变变量量时时,其其对对应应实实参参可可以以是是指针变量或存储单元地址。指针变量或存储单元地址。函函数数形形参参为为指指针针变变量量,用用指指针针变变量量或或变变量量地地址址作实参作实参例例8.3编编写写一一个个交交换换两两个个变变量量的的函函数数,在在主主程程序序中中调用,实现两个变量值的交换。调用,实现两个变量值的交换。5/4/202423C语言程序设计教程第8章 指针程序如下:#includemain()inta,b;int*pa,*pb;voidswap(int*p1,int*p2);scanf(%d%d,&a,&b);pa=&a;pb=&b;swap(pa,pb);printf(na=%d,b=%dn,a,b);或:swap(&a,&b);voidswap(int*p1,int*p2)inttemp;temp=*p1;*p1=*p2;*p2=temp;5/4/202424C语言程序设计教程第8章 指针分析下面程序的正确性:#includemain()inta,b;int*pa,*pb;voidswap(int*p1,int*p2);scanf(%d%d,&a,&b);pa=&a;pb=&b;swap(pa,pb);printf(na=%d,b=%dn,a,b);voidswap(int*p1,int*p2)int*temp;*temp=*p1;*p1=*p2;*p2=*temp;5/4/202425C语言程序设计教程第8章 指针#includemain()inta,b;int*pa,*pb;voidswap(int*p1,int*p2);scanf(%d%d,&a,&b);pa=&a;pb=&b;swap(pa,pb);printf(na=%d,b=%dn,*pa,*pb);voidswap(int*p1,int*p2)int*temp;temp=p1;p1=p2;p2=temp;注意:不能企图通过改变指针形参的值而使指针实参的值改变5/4/202426C语言程序设计教程第8章 指针7.2.2指针函数指针函数指针函数:是指返回值为指针的函数是指返回值为指针的函数指针函数的定义形式:指针函数的定义形式:类型标示符类型标示符 *函数名(参数)函数名(参数)例如:例如:intint*fun(intfun(int a,inta,int b)b)函数体语句函数体语句 在函数体中有返回指针或地址的语句,形如在函数体中有返回指针或地址的语句,形如:return(&变量名变量名);或或return(指针变量指针变量);并且返回值的类型要与函数类型一致。并且返回值的类型要与函数类型一致。5/4/202427C语言程序设计教程第8章 指针例7.3 分析如下程序main()inta,b,*p;int*max(intx,inty);scanf(“%d,%d”,&a,&b);p=max(a,b);printf(“max=%d”,*p);int*max(intx,inty)ifxy)return(&x);elsereturn(&y);5/4/202428C语言程序设计教程第8章 指针7.2.3 指向函数的指针一个函数包括一组指令序列,存储在某一段内存中,这段内一个函数包括一组指令序列,存储在某一段内存中,这段内存空间的起始地址称为存空间的起始地址称为函数的入口地址函数的入口地址称函数入口地址为称函数入口地址为函数的指针函数的指针。函数名函数名函数名函数名代表函数的入口地址代表函数的入口地址可以定义一个指针变量,其值等于该函数的入口地址,指向可以定义一个指针变量,其值等于该函数的入口地址,指向这个函数,这样通过这个指针变量也能调用这个函数。这种指针这个函数,这样通过这个指针变量也能调用这个函数。这种指针变量称为变量称为指向函数的指针变量指向函数的指针变量。定义指向函数的指针变量的一般形式为定义指向函数的指针变量的一般形式为:类型标识符(类型标识符(*指针变量名指针变量名)()();例如:例如:int(*p)();/*指针变量指针变量p可以指向一个整型函数可以指向一个整型函数*/float(*q)();/*指针变量指针变量q可以指向一个浮点型函数可以指向一个浮点型函数*/5/4/202429C语言程序设计教程第8章 指针刚定义的指向函数的指针变量,亦象其它指针变量一样要刚定义的指向函数的指针变量,亦象其它指针变量一样要赋以地址值才能引用。当将某个函数的入口地址赋给指向函数赋以地址值才能引用。当将某个函数的入口地址赋给指向函数的指针变量,就可用该指针变量来调用所指向的函数的指针变量,就可用该指针变量来调用所指向的函数给函数指针赋初值:将函数名(函数的入口地址值)赋给给函数指针赋初值:将函数名(函数的入口地址值)赋给指针变量指针变量例如例如intm,(*p)();intmax(inta,intb);则可以则可以p=max;/*p指向函数指向函数max()*/指针调用函数的指针调用函数的一般形式为一般形式为:(*指针变量指针变量)()(实参表实参表);如上例如上例:m=(*p)(12,22);/*比较比较m=max(12,22);*/5/4/202430C语言程序设计教程第8章 指针注意:用用函函数数指指针针调调用用函函数数是是间间接接调调用用,没没有有参参数数类类型型说说明明,C编编译译系系统统也也无无法法进进行行类类型型检检查查,因因此此,在在使使用用这这种种形形式式调调用用函函数数时时要要特特别别小小心心。实实参参一一定定要要和和指指针针所所指指函函数数的形参类型一致。的形参类型一致。函数指针可以作为函数参数,此时,当函数指针每次函数指针可以作为函数参数,此时,当函数指针每次指向不同的函数时,可执行不同的函数来完成不同的功指向不同的函数时,可执行不同的函数来完成不同的功能能例例 7.47.4 函数函数max()max()用来求一维数组的元素的最大值,在用来求一维数组的元素的最大值,在主调函数中用函数名调用该函数与用函数指针调用该函主调函数中用函数名调用该函数与用函数指针调用该函数来实现。数来实现。5/4/202431C语言程序设计教程第8章 指针程序如下:#includestdio.h#defineM8main()floatsumf,sump;floataM=11,2,-3,4.5,5,69,7,80;float(*p)();floatmax(floata,intn);p=max;sump=(*p)(a,M);sumf=max(a,M);printf(sump=%.2fn,sump);printf(sumf=%.2fn,sumf);floatmax(floata,intn)intk;floats;s=a0;for(k=0;kn;k+)if(sak)s=ak;returns;程序运行结果:sump=80.00sumf=80.005/4/202432C语言程序设计教程第8章 指针指向函数的指针的使用步骤:(1)定义一个指向函数的指针变量,形如:float(*p)();(2)为函数指针赋值,格式如下:p=函数名;注意:赋值时只需给出函数名,不要带参数。(3)通过函数指针调用函数,调用格式如下:s=(*p)(实参);5/4/202433C语言程序设计教程第8章 指针7.3 指针与数组7.3.1指向一维数组的指针指向一维数组的指针数组名是一个常量指针,它的值为该数组的首地址1.指指向向数数组组的的指指针针的的定定义义方方法法与与指指向向基基本本类类型型变变量量的指针的定义方法相同的指针的定义方法相同,例如:inta10=1,3,5,7,9;int*p;p=&a2;(把数组元素a2的地址赋给指针变量p)p=a;(把数组的首地址赋给指针变量p)5/4/202434C语言程序设计教程第8章 指针C语言规定:数组名代表数组首地址,是一个地址常量。因此,下面两个语句等价:p=&a0;p=a;在定义指针变量的同时可赋初值:inta10,*p=&a0;(或int*p=a;)等价于:int*p;p=&a0;两句。5/4/202435C语言程序设计教程第8章 指针指向数组的指针变量p&a013151719pa0:a9p9a+0p+1或a+1p+9或a+9*(a+9)或*(p+9)5/4/202436C语言程序设计教程第8章 指针2.通过指针引用数组元素*p=5;表示对p当前所指的数组元素赋以一个值5。C规定:p+1指向数组的下一元素(而不是将p值简单地加1)。p+1意味着使p的原值(地址)加d个字节(d为一个数组元素所占的字节数)。如果p的初值为&a0,则:(1)p+i和a+i就是ai的地址,或者说它们指向a数组的第i个元素(见下页图)。(2)*(p+i)或*(a+i)是p+i或a+i所指向的数组元素,即ai。(3)指向数组的指针变量也可以带下标,如pi与*(p+i)、ai等价。5/4/202437C语言程序设计教程第8章 指针*(p+i)a数组a0a1a2aia9pp+1,a+1p+i,a+ip+9,a+9综上所述,引用一个数组元素有二法:(1)下标法:如ai形式;(2)指针法:如*(a+i)或*(p+i)。其中a是数组名,p是指向数组的指针变量,其初值p=a。5/4/202438C语言程序设计教程第8章 指针main()inta10;inti;for(i=0;i10;i+)scanf(%d,&ai);printf(n);for(i=0;i10;i+)printf(%d,ai);例7.5 用三种方法输出数组全部元素。(1)下标法5/4/202439C语言程序设计教程第8章 指针main()inta10;inti;for(i=0;i10;i+)scanf(%d,&ai);printf(n);for(i=0;i10;i+)printf(%d,*(a+i);(2)通过数组名计算数组元素地址通过数组名计算数组元素地址,输出元素的值输出元素的值5/4/202440C语言程序设计教程第8章 指针(3)用指针变量指向数组元素main()inta10;int*p,i;for(i=0;i10;i+)scanf(%d,&ai);printf(n);for(p=a;p(a+10);p+)printf(%d,*p);三种方法的比较三种方法的比较:用下标法下标法比较直观,能直接知道是第几个元素;而用指针法指针法则执行效率更高。5/4/202441C语言程序设计教程第8章 指针使用指针变量时,应注意:(1)指针变量可实现使本身的值改变。P+合法;但a+不合法(a是数组名,代表数组首地址,在程序运行中是固定不变的。)(2)要注意指针变量的当前值。main()inta10;int*p,i;p=a;for(;pa+10;p+)scanf(%d,p);printf(n);for(;p(a+10);p+)printf(%d,*p);不能&p增加:p=a;5/4/202442C语言程序设计教程第8章 指针(3)*p+相当于*(p+),因为*与+优先级相同,且结合方向从右向左,其作用是先获得p指向变量的值,然后执行p=p+1;(4)*(p+)与*(+p)意义不同,后者是先p=p+1,再获得p指向的变量值。若p=a,则输出*(p+)是先输出a0,再让p指向a;输出*(+p)是先使p指向a,再输出p所指的a。(5)(*p)+表示的是将p指向的变量值+5/4/202443C语言程序设计教程第8章 指针例:编写程序,定义一个含有15个元素的数组,并编写函数分别完成以下操作:(1)调用C库函数中的随机函数给所有元素赋以050之间的随机数;(2)输出数组中的元素的值(3)按顺序对每隔三个数求一个和数,并传回主函数;(4)最后输出所有求出的和值5/4/202444C语言程序设计教程第8章 指针#include“stdlib.h”#defineSIZE15#defineN5main()intxSIZE,wN;getrand(x,SIZE);priarr(x,SIZE);getsum(x,w,SIZE);priarr(w,N);getrand(int*a,intn)inti;for(i=0;in;i+)ai=random(50);getsum(int*a,int*b,intn)inti,j,sum;for(sum=0,i=0,j=0;in;i+)for(sum=0,i=0,j=0;in;i+)sum+=ai;if(i+1)%3=0)bj=sum;sum=0;j+;pararr(int*a,intn)inti;for(i=0;in;i+)printf(“%5d”,ai);if(i+1)%5=0)printf(“n”);5/4/202445C语言程序设计教程第8章 指针作业:1.编写函数,其功能是对传送过来的的两个整数求出和值和差值,并通过形参传送回调用函数.2.编写程序,对传送过来的三个数选出最大和最小数,并通过形参传回调用函数.3.将数组中的数按颠倒的顺序重新存放.在操作时只能借助一个临时的存储单元而不得另外开辟数组.4.输入十个整数,将其中最小的数与第一个数对换,把最大的数与最后一个数对换.写三个函数(1)输入十个数(2)进行处理(3)输出十个数5.附加题:w数组中存放n个数据,编写函数删除下标为k的元素的值.5/4/202446C语言程序设计教程第8章 指针7.3.2 二维数组与多维数组的指针表示法 1.二维数组的地址二维数组的地址设有一个二维数组a,它有三行四列:inta33;数组名a:代表整个二维数组的首地址,也就是第0行的首地址。a+i:代表第i行的首地址。(见下页图)5/4/202447C语言程序设计教程第8章 指针数组名a代表整个二维数组的首地址:a0a1a2a=19173111971523上图 a数组包含三个元素:a0,a1,a2.而每个元素又是一个一维数组,它包含3个元素(即3个列元素),如:a0又包含:a00,a01,a02,a03.5/4/202448C语言程序设计教程第8章 指针一维数组名ai:代表第i行的首地址,即第i行中第0列元素的地址(既&ai0)。ai+j:代表第i行中的第j个元素的地址,即为&aij。注意地址变化的单位数值在不同的场合的实际字节数是不同的:“a+1”中的“1”实际代表数组中一行元素所占的总字节数;“ai+1”中的“1”代表数组中一个元素所占的字节数。5/4/202449C语言程序设计教程第8章 指针a+2a数组a0a1a2aa+1(2000)(2008)(2016)a代表第0行的首地址,a+1代表第1行的首地址,a+2代表第2行的首地址.每行存放4个整型数据(即1个元素占2个字节),因此,这里+1的含义是:+4*2=+8个字节.5/4/202450C语言程序设计教程第8章 指针2000120023200452006720089201011201213201415201617201819202021202223aa+1a+2a0+1 a0+2 a0+3行地址,元素地址的表示法5/4/202451C语言程序设计教程第8章 指针例例7.6 7.6 用指针表示法输出二维数组的各元素。用指针表示法输出二维数组的各元素。#includemain()staticinta23=0,1,2,3,4,5;intk,j,*p;for(j=0;j2;j+)/*方式方式1*/for(k=0;k3;k+)printf(%5d,*(aj+k);/*aj是是j行首地址,行首地址,aj+k是是j行行k列元素的地址列元素的地址*/putchar(n);putchar(n);5/4/202452C语言程序设计教程第8章 指针接上页:for(j=0;j2;j+)/*方式方式2*/for(k=0;k3;k+)printf(%5d,*(*(a+j)+k);/*(a+j)是是j行行首首地地址址,*(a+j)+k是是j行行k列列元元素素的的地地址址*/putchar(n);p=a0;/*p指向数组的第一个元素*/for(j=0;j2;j+)/*方式方式3*/for(k=0;k3;k+)printf(%5d,*(p+);/*输出输出p所指示的元素所指示的元素*/putchar(n);5/4/202453C语言程序设计教程第8章 指针输出的结果是:0345034503455/4/202454C语言程序设计教程第8章 指针2.指向二维数组的指针变量指向二维数组的指针变量有两种情况有两种情况:一是直接指向数组元素的指针变量;一是直接指向数组元素的指针变量;二是指向一个含有二是指向一个含有m个元素的一维数组。个元素的一维数组。这两种不同形式的指针变量,其使用方法不同。这两种不同形式的指针变量,其使用方法不同。(1)指向数组元素的指针变量指向数组元素的指针变量,即简单指针变量,如:即简单指针变量,如:int*p,a34;p=&a12;(2)指向由指向由m个元素组成的一维数组的指针变量个元素组成的一维数组的指针变量定义形式:类型类型 (*指针变量)指针变量)元素个数元素个数 如:int(*p)4;5/4/202455C语言程序设计教程第8章 指针注意:(1)int(*p)4;定义一个指针变量p,p指向包含4个元素的一维数组。(2)p+i与*(p+i)的区别:p+i是指向第i行的指针(第i行的首地址);*(p+i)是指向第i行第1个元素的地址;两者数值相等,但含义不同:两者数值相等,但含义不同:p+i p+i 的增值将以行的增值将以行长为单位,长为单位,而而*(*(p+i)p+i)增值增值将以元素长度为单位。将以元素长度为单位。5/4/202456C语言程序设计教程第8章 指针即:p+i+1将指向第i行再下一行的首地址,而*(p+i)+1将指向第i行首元素的下一个元素地址。(见下图)设inta34,(*p)4;p=a;P,aP+1P+2a0a1a2如果p先指向a0,则p+1不是指向a01,而是指向a15/4/202457C语言程序设计教程第8章 指针例例7.7main()staticinta34=1,3,5,7,9,11,13,15,17,19,21,23;int(*p)4,i,j;p=a;scanf(i=%d,j=%d,&i,&j);printf(a%d%d=%dn,i,j,*(*(p+i)+j);5/4/202458C语言程序设计教程第8章 指针二维数组的指针作为函数参数有一个班,3个学生,各学4门课,计算总平均成绩,以及第n个学生的成绩main()floatscore34=65,67,70,60,80,87,90,98;average(*score,12);/求序号为2的学生成绩search(score,2);voidaverage(float*p,intn)float*p_end;floatsum=0,aver;p_end=p+n-1;for(;p=9_end;p+)sum=sum+(*p);aver=sum/n;printf(“%fn”,aver);voidsearch(float(*p)4,intn)inti;for(i=0;i=n)p+=n-1;/*指针指到要复制的第一个字符指针指到要复制的第一个字符*/5/4/202463C语言程序设计教程第8章 指针for(;*p!=0;p+,q+)*q=*p;*q=0;/*字符串以0结尾*/printf(Stringa:%sn,a);printf(Stringb:%sn,b);输入:3输出:computermputer考虑一下,若输出语句改为如下语句会如何?printf(“string a is:%sn”,p);printf(“string b is%sn”,q);5/4/202464C语言程序设计教程第8章 指针7.4.2字符串数组字符串数组字符串数组:是指数组中的每个元素都是一个存放字符串的数组。字符串数组可以用一个二维字符数组来存储字符串数组可以用一个二维字符数组来存储。例如:charlanguge310;数组的第一个下标决定字符串的个数,第二个下标是字符串的最大长度(实际最多9个字符,0占一位置)。可以对字符串数组赋初值。例如:char languge3 10;=“Basic”,“c+”,“pascal”5/4/202465C语言程序设计教程第8章 指针7.5 7.5 指针数组与命令行参数指针数组与命令行参数7.5.1 7.5.1 指针数组指针数组指针数组:是指针变量的集合。即它的每一个元素都是指针变量,且都具有相同的存储类别和指向相同的数据类型。指针数组的定义形式为:类型标识符 *数组名数组长度说明;例如:int*p10;由于比*的优先级高,因此p先与10结合成p10,而p10正是数组的定义形式,共有10个元素。最后p10与*结合,表示它的各元素可以指向一个整型变量。5/4/202466C语言程序设计教程第8章 指针指针数组广泛应用于对字符串的处理指针数组广泛应用于对字符串的处理例如有定义:例如有定义:char*p3;定定义义了了一一个个具具有有三三个个元元素素p0,p1,p2的的指指针针数数组组。每个元素都可以指向一个字符数组,或字符串。每个元素都可以指向一个字符数组,或字符串。若利用数组初始化,则:若利用数组初始化,则:char*p3=“Basic”,“c+”,“pascal”;P0指向字符串指向字符串“Basic”;P0指向字符串指向字符串“c+”;P0指向字符串指向字符串“pascal”;5/4/202467C语言程序设计教程第8章 指针例7.9 字符指针数组的赋值#defineNULL0main()staticchara=“Fortran”;staticcharb=“COBOL”;staticcharc=“Pascal”;inti;char*p4;p0=a;p1=b;p2=c;p3=NULL;for(i=0;pi!=NULL;i+)printf(“Language%dis%sn”,i+1,pi);5/4/202468C语言程序设计教程第8章 指针例7.10 有若干本书,将书名按字典顺序排序#include#includemain()char*bname=ProgramminginANSIC,BASIC,VisualC+6.0Programming,TRUBOC2.0;inti,m;voidsort(char*name,int);m=sizeof(bname)/sizeof(char*);/*字符串个数字符串个数*/sort(bname,m);/*排序,改变指针的连接关系排序,改变指针的连接关系*/printf(n);for(i=0;im;i+)/*输出排序结果输出排序结果*/printf(%8s,bnamei);5/4/202469C语言程序设计教程第8章 指针voidsort(char*name,intn)/*选择排序*/char*t;inti,j,k;/*k记录每趟最小值下标*/for(i=0;in-1;i+)k=i;for(j=i+1;j0)k=j;/*第j个元素更小*/if(k!=i)/*最小元素是该趟的第一个元素则不需交换*/t=namei;namei=namek;namek=t;输出结果为:BASICProgramminginANSICTRUBOC2.0VisualC+6.0Programming5/4/202470C语言程序设计教程第8章 指针注意注意:(1)字符数组中每个元素可存放一个字符,而字符指针变量存放字符串首地址,而不是存放在字符指针变量中。(2)对字符数组,与普通数组一样,不能对其进行整体赋值,只能给各个元素赋值,而字符指针变量可以直接用字符串常量赋值。例如,若有如下定义:chara10;char*p;则语句a=”computer”;是非法的,因为数组名a是一个常量指针,不能对其赋值。只能对各个元素分别赋值:a0=c;a1=o;a2=m;a3=p;a7=r;但语句:p=”computer”;是合法的。5/4/202471C语言程序设计教程第8章 指针7.5.2 7.5.2 指针数组与命令行参数指针数组与命令行参数在操作系统命令状态下,可以输入程序或命令使其运行,称命命令令行行状状态态。输入的命令(或运行程序)及该命令(或程序)所需的参数称为命令行参数命令行参数。如如:copyfdfscopy是文件拷贝命令,是文件拷贝命令,fd、fs是命令行参数。是命令行参数。main函数是可以有参数的,但与普通函数不同。带形参的main()函数的一般形式是:main(intint argcargc,char*,char*argvargv)形参argc记录命令行中字符串的个数,argv是一个字符型指针数组,每一个元素顺序指向命令行中的一个字符串。5/4/202472C语言程序设计教程第8章 指针1.main()1.main()函数的形参与实参函数的形参与实参mainmain()函函数数由由系系统统自自动动调调用用,而而不不是是被被程程序序内内部部的的其其它它函函数数调调用用,main()main()函函数数所所需需的的实实参参不不可可能能由由程程序序内内部部得得到到,而是由系统传送。而是由系统传送。main()函数所需的实参与形参的传递方式也与一般函数的参数传递不同,实参是在命令行与程序名一同输入,程序名和各实际参数之间都用空格分隔。格式为:执行程序名 参数1 参数2 参数n形形参参argc为为命命令令行行中中参参数数的的个个数数(包括执行程序名),其值大于或等于1,而不是象普通C语言函数一样接受第一个实参。形形参参argv是是一一个个指指针针数数组组,其其元元素素依依次次指指向向命命令令行行中中以以空空格格分分开的各字符串。开的各字符串。即:第一个指针argv0指向的是程序名字符串,argv1指向参数1,argv2指向参数2,argvn指向参数n。5/4/202473C语言程序设计教程第8章 指针2.2.命令行参数的传递示例命令行参数的传递示例例例7.11 7.11 分析下列程序,指出其执行结果,该程序命名为分析下列程序,指出其执行结果,该程序命名为exam.c,exam.c,经编译连接后生成的可执行程序为经编译连接后生成的可执行程序为exam.exeexam.exe#include#include main(int argc,char *argv )int i=0;printf(“argc=%dn”,argc);while (argc=1)printf(“n参数参数%d:%s”,i,*argv);i+;argc-;argv+;5/4/202474C语言程序设计教程第8章 指针输出结果:输出结果:argcargc=4=4参数参数0 0:examexam参数参数1 1:Turbo_cTurbo_c参数参数2 2:C+C+参数参数3 3:VcVc程程序序开开始始运运行行后后,系系统统将将命命令令行行中中字字符符串串个个数数送送argcargc,将将四四个个字字符符串串实实参参:examexam、Turbo_cTurbo_c、C+C+、VcVc的的首首地地址址分分别别传传给给字符指针数组元素字符指针数组元素argv0argv0、argv1argv1、argv2argv2、argv3argv3。若运行该程序时的命令行输入的是:若运行该程序时的命令行输入的是:exam Turbo_c C+exam Turbo_c C+VcVc5/4/202475C语言程序设计教程第8章 指针7.6 7.6 程序举例程序举例例8.12 输入一个十进制正整数,将其转换成二进制、八进制、十六进制数输出 分析:(1)将十进制数n转换成r进制数的方法是:n除以r取余数作为转换后的数的最低位。若商不为0,则商继续除以r,取余数作为次低位,以此类推,直到商为0为止。(2)对于十六进制数中大于9的六个数字是用A,B,C,D,E,F来表示。(3)所得余数序列转换成字符保存在字符数组a中。(4)字符0的ascii码是48,故余数09只要加上48就变成字符09了;余数中大于9的数1015要转换成字母,加 上 55就 转 换 成 A、B、C、D、E、F了。(5)由于求得的余数序列是低位到高位,而屏幕显示先显示高位,所以输出数组a时要反向进行。5/4/202476C语言程序设计教程第8章 指针#includestdio.hmain()inti,radix;longn;chara33;voidtrans10_2_8_16(charb,longm,intbase);printf(nInputradix(2,8,16):);/*输入转换基数*/scanf(%d,&radix);printf(nInputapositiveinteger:);/*输入被转换的数*/scanf(%ld,&n);trans10_2_8_16(a,n,radix);for(i=strlen(a)-1;i=0;i-)/*逆向输出字符串*/printf(%c,*(a+i);/*(a+i)即ai*/puts(n);5/4/202477C语言程序设计教程第8章 指针voidtrans10_2_8_16(char*p,longm,intbase)intr;while(m0)r=m%base;/*求余数*/if(r10)*p=r+48;/*小于10的数转换成字符后送p指向的元素*/else*p=r+55;/*数1015转换成AF后送p指向的元素*/m=m/base;p+;/*指针下移*/*p=0;/*在最后加上字符串结束标志*/输入:输入:Inputradix(2,8,16):16Inputapositiveinteger:435678输出:输出:6A5DE5/4/202478C语言程序设计教程第8章 指针作业:1.编写函数把数组中的所有奇数放在另一个数组中返回.2.编写程序通过调用随机函数给5*6的二维数组赋1040的整数,求出数组每行元素的平均值.3.有n个人围成一圈,顺序排号.从第1个人开始报数(从1到3报数),凡报到3的人退出圈子,问最后留下的是原来的第几号的那位.4.编写一程序,输入月份号,输出该月的英文月名.例如,输入”3”,则输出”March”,要求用指针数组处理.5/4/202479C语言程序设计教程
展开阅读全文
相关资源
相关搜索

最新文档


当前位置:首页 > 商业管理 > 营销创新


copyright@ 2023-2025  zhuangpeitu.com 装配图网版权所有   联系电话:18123376007

备案号:ICP2024067431-1 川公网安备51140202000466号


本站为文档C2C交易模式,即用户上传的文档直接被用户下载,本站只是中间服务平台,本站所有文档下载所得的收益归上传人(含作者)所有。装配图网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对上载内容本身不做任何修改或编辑。若文档所含内容侵犯了您的版权或隐私,请立即通知装配图网,我们立即给予删除!