C语言第十章指针.

整理文档很辛苦,赏杯茶钱您下走!

免费阅读已结束,点击下载阅读编辑剩下 ...

阅读已结束,您可以下载文档离线阅读编辑

资源描述

第十章§10.1地址和指针的概念为了说清楚什么是指针,必须弄清楚数据在内存中是如何存储的,又是如何读取的。内存区的每一个字节有一个编号,这就是“地址”。如果在程序中定义了一个变量,在对程序进行编译时,系统就会给这个变量分配内存单元。1、按变量地址存取变量值的方式称为“直接访问”方式printf(″%d″,i);scanf(″%d″,&i);k=i+j;例如:另一种存取变量值的方式称为“间接访问”的方式。即,将变量i的地址存放在另一个变量中。在C语言中,指针是一种特殊的变量,它是存放地址的。假设我们定义了一个指针变量i_pointer用来存放整型变量的地址,它被分配地址为(3010)、(3011)的两个字节。可以通过语句:i_pointer=&i;将i的地址(2000)存放到i_pointer中。这时,i_pointer的值就是(2000),即变量i所占用单元的起始地址。要存取变量i的值,可以采用间接方式:先找到存放“i的地址”的变量i_pointer,从中取出i的地址(2000),然后到2000、2001字节取出i的值(3)。一个变量的地址称为该变量的“指针”。例如,地址2000是变量i的指针。如果有一个变量专门用来存放另一变量的地址(即指针),则它称为“指针变量”。上述的i_pointer就是一个指针变量。指针和指针变量的定义:指针变量的值(即指针变量中存放的值)是地址(即指针)。请区分“指针”和“指针变量”这两个概念。§10.2变量的指针和指向变量的指针变量10.2.1定义一个指针变量定义指针变量的一般形式为基类型*指针变量名;下面都是合法的定义:float*pointer_3;//pointer_3是指向float型变量的指针变量char*pointer_4;//pointer_4是指向字符型变量的指针变量可以用赋值语句使一个指针变量得到另一个变量的地址,从而使它指向一个该变量。如:pointer_1=&i;pointer_2=&j;在定义指针变量时要注意两点:(1)指针变量前面的“*”,表示该变量的类型为指针型变量。例:float*pointer_1;指针变量名是pointer_1,而不是*pointer_1。(2)在定义指针变量时必须指定基类型。需要特别注意的是,只有整型变量的地址才能放到指向整型变量的指针变量中。下面的赋值是错误的∶floata;int*pointer_1;pointer_1=&a;/*将float型变量的地址放到指向整型变量的指针变量中,错误*/10.2.2指针变量的引用请牢记,指针变量中只能存放地址(指针),不要将一个整数(或任何其他非地址类型的数据)赋给一个指针变量。例10.1通过指针变量访问整型变量#includestdio.hvoidmain(){inta,b;int*pointer_1,*pointer_2;a=100;b=10;pointer_1=&a;/*把变量a的地址赋给pointer_1*/pointer_2=&b;/*把变量b的地址赋给pointer_2*/printf(″%d,%d\n″,a,b);printf(″%d,%d\n″,*pointer_1,*pointer_2);}对“&”和“*”运算符说明:如果已执行了语句pointer_1=&a;(1)&*pointer_1的含义是什么?“&”和“*”两个运算符的优先级别相同,但按自右而左方向结合,因此先进行*pointer_1的运算,它就是变量a,再执行&运算。因此,&*pointer_1与&a相同,即变量a的地址。如果有pointer_2=&*pointer_1;它的作用是将&a(a的地址)赋给pointer_2,如果pointer_2原来指向b,经过重新赋值后它已不再指向b了,而指向了a。(2)*&a的含义是什么?先进行&a运算,得a的地址,再进行*运算。即&a所指向的变量,也就是变量a。*&a和*pointer_1的作用是一样的,它们都等价于变量a。即*&a与a等价。(3)(*pointer_1)++相当于a++。注意括号是必要的,如果没有括号,就成为了*pointer_1++,从附录可知:++和*为同一优先级别,而结合方向为自右而左,因此它相当于*(pointer_1++)。由于++在pointer_1的右侧,是“后加”,因此先对pointer_1的原值进行*运算,得到a的值,然后使pointer_1的值改变,这样pointer_1不再指向a了。例10.2输入a和b两个整数,按先大后小的顺序输出a和b。#includestdio.hvoidmain(){int*p1,*p2,*p,a,b;scanf(″%d,%d″,&a,&b);p1=&a;p2=&b;if(a<b){p=p1;p1=p2;p2=p;}printf(″a=%d,b=%d\n\n″,a,b);printf(″max=%d,min=%d\n″,*p1,*p2);}运行情况如下:5,9↙a=5,b=9max=9,min=5当输入a=5,b=9时,由于a<b,将p1和p2交换。交换前的情况见图(a),交换后见图(b)。10.2.3指针变量作为函数参数例10.3对输入的两个整数按大小顺序输出#includestdio.hvoidmain(){voidswap(int*p1,int*p2);inta,b;int*pointer_1,*pointer_2;scanf(″%d,%d″,&a,&b);pointer_1=&a;pointer_2=&b;if(a<b=swap(pointer_1,pointer_2);printf(″\n%d,%d\n″,a,b);}voidswap(int*p1,int*p2){inttemp;temp=*p1;*p1=*p2;*p2=temp;}例10.4输入a、b、c3个整数,按大小顺序输出#includestdio.hvoidmain(){voidexchange(int*q1,int*q2,int*q3);inta,b,c,*p1,*p2,*p3;scanf(″%d,%d,%d″,&a,&b,&c);p1=&a;p2=&b;p3=&c;exchange(p1,p2,p3);printf(″\n%d,%d,%d\n″,a,b,c);}voidexchange(int*q1,int*q2,int*q3){voidswap(int*pt1,int*pt2);if(*q1<*q2)swap(q1,q2);if(*q1<*q3)swap(q1,q3);if(*q2<*q3=swap(q2,q3);}voidswap(int*pt1,int*pt2){inttemp;temp=*pt1;*pt1=*pt2;*pt2=temp;}§10.3数组与指针一个变量有地址,一个数组包含若干元素,每个数组元素都在内存中占用存储单元,它们都有相应的地址。指针变量既然可以指向变量,当然也可以指向数组元素(把某一元素的地址放到一个指针变量中)。所谓数组元素的指针就是数组元素的地址。定义一个指向数组元素的指针变量的方法,与以前介绍的指向变量的指针变量相同。例如:inta[10];(定义a为包含10个整型数据的数组)*p;(定义p为指向整型变量的指针变量)应当注意,如果数组为int型,则指针变量的基类型亦应为int型。10.3.1指向数组元素的指针对该指针变量赋值:p=&a[0];把a[0]元素的地址赋给指针变量p。也就是使p指向a数组的第0号元素,如图:10.3.2通过指针引用数组元素引用一个数组元素,可以用:(1)下标法,如a[i]形式;(2)指针法,如*(a+i)或*(p+i)。其中a是数组名,p是指向数组元素的指针变量,其初值p=a。例10.5输出数组中的全部元素假设有一个a数组,整型,有10个元素。要输出各元素的值有三种方法:(1)下标法#includestdio.hvoidmain(){inta[10];inti;for(i=0;i<10;i++)scanf(″%d″,&a[i]);printf(″\n″);for(i=0;i<10;i++)printf(″%d″,a[i]);}(2)通过数组名计算数组元素地址,找出元素的值。#includestdio.hvoidmain(){inta[10];inti;for(i=0;i<10;i++)scanf(″%d″,&a[i]);printf(″\n″);for(i=0;i<10;i++)printf(″%d″,*(a+i));}(3)用指针变量指向数组元素。#includestdio.hvoidmain(){inta[10];int*p,i;for(i=0;i<10;i++)scanf(″%d″,&a[i]);printf(″\n″);for(p=a;p<(a+10);p++)printf(″%d″,*p);}例10.6通过指针变量输出a数组的10个元素。有人编写出以下程序:#includestdio.hvoidmain(){int*p,i,a[10];p=a;for(i=0;i<10;i++)scanf(″%d″,p++);printf(″\n″);for(i=0;i<10;i++,p++)printf(″%d″,*p);}这个程序乍看起来好像没有什么问题。有的人即使已被告知此程序有问题,还是找不出它有什么问题。我们先看一下运行情况:1234567890↙22153234003003625202116318259823728483显然输出的数值并不是a数组中各元素的值解决这个问题的办法,只要在第二个for循环之前加一个赋值语句:p=a;#includestdio.hvoidmain(){int*p,i,a[10];p=a;for(i=0;i<10;i++)scanf(″%d″,p++);printg(″\n″);p=a;for(i=0;i<10;i++,p++)printf(″%d″,*p);}10.3.3用数组名作函数参数在第8章8.7节中介绍过可以用数组名作函数的参数如:voidmain(){if(intarr[],intn);intarray[10];┇f(array,10);┇}voidf(intarr[],intn){┇}f(intarr[],intn)但在编译时是将arr按指针变量处理的,相当于将函数f的首部写成f(int*arr,intn)以上两种写法是等价的。需要说明的是:C语言调用函数时虚实结合的方法都是采用“值传递”方式,当用变量名作为函数参数时传递的是变量的值,当用数组名作为函数参数时,由于数组名代表的是数组首元素地址,因此传递的值是地址,所以要求形参为指针变量。例10.7将数组a中n个整数按相反顺序存放#includestdio.hvoidmain(){voidinv(intx[],intn);inti,a[10]={3,7,9,11,0,6,7,5,4,2};printf(″Theoriginalarray:\n″);for(i=0;i<10;i++)printf(″%d,″,a[i]);printf(″\n″);inv(a,10);printf(″Thearrayhasbeeninverted:\n″);for(i=0;i<10;i++)printf(″%d,″,a[i]);printf(″\n″);}voidinv(intx[],intn)/*形参x是数组名*/{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;}运行情况如下:Theoriginalarray:3,7,9,11,0,6,7,5,4,2,Thearrayhasbeeninverted:2,4,5,7,6,0,11,9,7,3,#includestdio.hvoidmain(){voidinv(int*x,intn);inti

1 / 110
下载文档,编辑使用

©2015-2020 m.777doc.com 三七文档.

备案号:鲁ICP备2024069028号-1 客服联系 QQ:2149211541

×
保存成功