正文

C语言初学者入门讲座 第十二讲 多维数组的指针变量2007-05-04 17:14:00

【评论】 【打印】 【字体: 】 本文链接:http://blog.pfan.cn/lym51/25541.html

分享到:

C语言初学者入门讲座 第十二讲 多维数组的指针变量   一、多维数组地址的表示方法   设有整型二维数组a[3][4]如下:   0 1 2 3   4 5 6 7   8 9 10 11   设数组a的首地址为1000,各下标变量的首地址及其值如图所示。 在前面曾经介绍过, C语言允许把一个二维数组分解为多个一维数组来处理。因此数组a可分解为三个一维数组,即a[0],a[1],a[2]。每一个一维数组又含有四个元素。例如a[0]数组,含有a[0][0],a[0][1],a[0][2],a[0][3]四个元素。数组及数组元素的地址表示如下:a是二维数组名,也是二维数组0行的首地址,等于1000。a[0]是第一个一维数组的数组名和首地址,因此也为 1000。*(a+0)或*a是与a[0]等效的, 它表示一维数组a[0]0 号元素的首地址。也为1000。&a[0][0]是二维数组a的0行0列元素首地址,同样是1000。因此,a,a[0],*(a+0),*a,&a [0][0]是相等的。同理,a+1是二维数组1行的首地址,等于1008。a[1]是第二个一维数组的数组名和首地址,因此也为1008。 &a[1][0]是二维数组a的1行0列元素地址,也是1008。因此a+1,a[1],*(a+1),&a[1][0]是等同的。由此可得出:a+i,a[i],*(a+i),&a[i][0]是等同的。此外,&a[i]和a[i]也是等同的。因为在二维数组中不能把&a[i]理解为元素a[i]的地址,不存在元素a[i]。   C语言规定,它是一种地址计算方法,表示数组a第i行首地址。由此,我们得出:a[i],&a[i],*(a+i)和a+i也都是等同的。另外,a[0]也可以看成是a[0]+0是一维数组a[0]的0号元素的首地址,而a[0]+1则是a[0]的1号元素首地址,由此可得出a[i]+j则是一维数组a[i]的j号元素首地址,它等于&a[i][j]。由a [i]=*(a+i)得a[i]+j=*(a+i)+j,由于*(a+i)+j是二维数组a的i行j列元素的首地址。该元素的值等于*(*(a+i)+ j)。 [Explain] #define PF "%d,%d,%d,%d,%d,/n" main(){ static int a[3][4]={0,1,2,3,4,5,6,7,8,9,10,11}; printf(PF,a,*a,a[0],&a[0],&a[0][0]); printf(PF,a+1,*(a+1),a[1],&a[1],&a[1][0]); printf(PF,a+2,*(a+2),a[2],&a[2],&a[2][0]); printf("%d,%d/n",a[1]+1,*(a+1)+1); printf("%d,%d/n",*(a[1]+1),*(*(a+1)+1)); }   二、多维数组的指针变量   把二维数组a 分解为一维数组a[0],a[1],a[2]之后,设p为指向二维数组的指针变量。可定义为: int (*p)[4] 它表示p是一个指针变量,它指向二维数组a 或指向第一个一维数组a[0],其值等于a,a[0],或&a[0][0]等。而p+i则指向一维数组a[i]。从前面的分析可得出*(p+i) +j是二维数组i行j 列的元素的地址,而*(*(p+i)+j)则是i行j列元素的值。   二维数组指针变量说明的一般形式为:类型说明符 (*指针变量名)[长度] 其中“类型说明符”为所指数组的数据类型。“*”表示其后的变量是指针类型。 “长度”表示二维数组分解为多个一维数组时,一维数组的长度,也就是二维数组的列数。应注意“(*指针变量名)”两边的括号不可少,如缺少括号则表示是指针数组(本章后面介绍),意义就完全不同了。 [Explain] main(){ static int a[3][4]={0,1,2,3,4,5,6,7,8,9,10,11}; int(*p)[4]; int i,j; p=a; for(i=0;i<3;i++) for(j=0;j<4;j++) printf("%2d ",*(*(p+i)+j)); }   'Expain字符串指针变量的说明和使用字符串指针变量的定义说明与指向字符变量的指针变量说明是相同的。只能按对指针变量的赋值不同来区别。对指向字符变量的指针变量应赋予该字符变量的地址。如: char c,*p=&c;表示p是一个指向字符变量c的指针变量。而: char *s="C Language";则表示s是一个指向字符串的指针变量。把字符串的首地址赋予s。   请看下面一例。 main(){ char *ps; ps="C Language"; printf("%s",ps); }   运行结果为:   C Language   上例中,首先定义ps是一个字符指针变量,然后把字符串的首地址赋予ps(应写出整个字符串,以便编译系统把该串装入连续的一块内存单元),并把首地址送入ps。程序中的: char *ps;ps="C Language";等效于: char *ps="C Language";输出字符串中n个字符后的所有字符。 main(){ char *ps="this is a book"; int n=10; ps=ps+n; printf("%s/n",ps); }   运行结果为:   book 在程序中对ps初始化时,即把字符串首地址赋予ps,当ps= ps+10之后,ps指向字符“b”,因此输出为"book"。 main(){ char st[20],*ps; int i; printf("input a string:/n"); ps=st; scanf("%s",ps); for(i=0;ps[i]!='/0';i++) if(ps[i]=='k'){ printf("there is a 'k' in the string/n"); break; } if(ps[i]=='/0') printf("There is no 'k' in the string/n"); }   本例是在输入的字符串中查找有无‘k’字符。 下面这个例子是将指针变量指向一个格式字符串,用在printf函数中,用于输出二维数组的各种地址表示的值。但在printf语句中用指针变量PF代替了格式串。 这也是程序中常用的方法。 main(){ static int a[3][4]={0,1,2,3,4,5,6,7,8,9,10,11}; char *PF; PF="%d,%d,%d,%d,%d/n"; printf(PF,a,*a,a[0],&a[0],&a[0][0]); printf(PF,a+1,*(a+1),a[1],&a[1],&a[1][0]); printf(PF,a+2,*(a+2),a[2],&a[2],&a[2][0]); printf("%d,%d/n",a[1]+1,*(a+1)+1); printf("%d,%d/n",*(a[1]+1),*(*(a+1)+1)); }   在下例是讲解,把字符串指针作为函数参数的使用。要求把一个字符串的内容复制到另一个字符串中,并且不能使用strcpy函数。函数cprstr的形参为两个字符指针变量。pss指向源字符串,pds指向目标字符串。表达式: (*pds=*pss)!=`/0' cpystr(char *pss,char *pds){ while((*pds=*pss)!='/0'){ pds++; pss++; } } main(){ char *pa="CHINA",b[10],*pb; pb=b; cpystr(pa,pb); printf("string a=%s/nstring b=%s/n",pa,pb); }   在上例中,程序完成了两项工作:一是把pss指向的源字符复制到pds所指向的目标字符中,二是判断所复制的字符是否为`/0',若是则表明源字符串结束,不再循环。否则,pds和pss都加1,指向下一字符。在主函数中,以指针变量pa,pb为实参,分别取得确定值后调用cprstr函数。由于采用的指针变量pa和pss,pb和pds均指向同一字符串,因此在主函数和cprstr函数中均可使用这些字符串。也可以把cprstr函数简化为以下形式: cprstr(char *pss,char*pds) {while ((*pds++=*pss++)!=`/0');}   即把指针的移动和赋值合并在一个语句中。 进一步分析还可发现`/0'的ASCⅡ码为0,对于while语句只看表达式的值为非0就循环,为0则结束循环,因此也可省去“!=`/0'”这一判断部分,而写为以下形式: cprstr (char *pss,char *pds) {while (*pdss++=*pss++);}   表达式的意义可解释为,源字符向目标字符赋值, 移动指针,若所赋值为非0则循环,否则结束循环。这样使程序更加简洁。简化后的程序如下所示。 cpystr(char *pss,char *pds){ while(*pds++=*pss++); } main(){ char *pa="CHINA",b[10],*pb; pb=b; cpystr(pa,pb); printf("string a=%s/nstring b=%s/n",pa,pb); }   使用字符串指针变量与字符数组的区别   用字符数组和字符指针变量都可实现字符串的存储和运算。 但是两者是有区别的。在使用时应注意以下几个问题:   1. 字符串指针变量本身是一个变量,用于存放字符串的首地址。而字符串本身是存放在以该首地址为首的一块连续的内存空间中并以‘/0’作为串的结束。字符数组是由于若干个数组元素组成的,它可用来存放整个字符串。   2. 对字符数组作初始化赋值,必须采用外部类型或静态类型,如: static char st[]={“C Language”};而对字符串指针变量则无此限制,如: char *ps="C Language";   3. 对字符串指针方式 char *ps="C Language";可以写为: char *ps; ps="C Language";而对数组方式: static char st[]={"C Language"};   不能写为: char st[20];st={"C Language"};   而只能对字符数组的各元素逐个赋值。   从以上几点可以看出字符串指针变量与字符数组在使用时的区别,同时也可看出使用指针变量更加方便。前面说过,当一个指针变量在未取得确定地址前使用是危险的,容易引起错误。但是对指针变量直接赋值是可以的。因为C系统对指针变量赋值时要给以确定的地址。因此, char *ps="C Langage";   或者 char *ps; ps="C Language";   都是合法的。 站内提供文章版权均属原创作者所有,如有侵权请发信告知后删除! 本类热门文章 ·linux shell 编程学习笔记 ·一个排序用的C++函数模板 ·请问外部变量和内部变量是什么意思? ·使用cppunit为c++工程做单元测试 ·怎么用C程序写出一个能复制自己的程序? ·为什么这里不能产生随机数01? ·在Linux下如何快速搭建安全的FTP服务器(vsftpd) ·connect错误 ·[C]指针练习-2 ·#define Extern后面怎么没有参数也行? ·看不懂strrchr(),请帮忙。 ·使用mtrace检查内存溢出 ·学习笔记——boot ·编写另一个printf ·一个很莱的C程序 ·APUE并没有列出所有的关于信号的函数!! ·怎么这个程序不能通过? ·缺少gets函数,怎么办? ·使用read()的一个例子 ·不明白系统调用的意思? ·一个进程同时拥有的线程数目有最大限制吗??ZT ·C函数的一个规范写法 ·LINUX KERNEL 配置编译中文指南1 ·从一段代码看常量折叠。 ·C语言--位运算补遗 (转载) ·高位优先和低位优先 ·C连MYSQL ·确认Buffer ·C语言笔记 ·请问cache 和buff的意思? ·扬声器 ·奇怪的返回:ask :ask ·不理解怎么能打印出所有的环境字符串?? ·怎么把阻塞的信号集赋值给新的阻塞集? ·关于sleep() AND alrm() ·请问怎么可以得到一个sock的buffer ? ·常用函数速查 ·[C]libpcap函数库说明 ·求无符号整数的奇偶校验码 ·使用Visual C++调试器调试[转]

阅读(2526) | 评论(0)


版权声明:编程爱好者网站为此博客服务提供商,如本文牵涉到版权问题,编程爱好者网站不承担相关责任,如有版权问题请直接与本文作者联系解决。谢谢!

评论

暂无评论
您需要登录后才能评论,请 登录 或者 注册