比如要定义(开辟内存)一个字符数组,10个学生,每个学生的名字的字符长度最多是20,可以这样定义:(当然更好的是用结构体定义)
char students_name[10][20]={"gbhgbh","baobao","huahua","",""};
char *ps = students_name; //指针ps指向字符数组第一个元素students_name[0][0]的地址,也就是数组的首地址
取第三个同学的名字(也是获取第三个同学名字的首地址)
1、用数组形式,students_name[2]
,用printf("%s",students_name[2]);
将打印出huahua
2、用指针形式,students_name+2
,用printf("%s",students_name+2);
将打印出huahua
。或者用printf("%s",*(students_name+2));
也打印出huahua
。
3、特别注意,此时的ps+2是指向整个多维数组的第3个元素,即"gbhgbh"中的'h',则用printf("%s",ps+2);
将打印出hgbh
,到字符串结束符停止打印。取第三个同学名字中的第5个字符
1、用数组形式,students_name[2][4]
,用printf("%c",students_name[2][4]);
将打印出u
。
2、用指针形式,*(*(students_name+2)+4)
,用printf("%c",*(*(students_name+2)+4));
将打印出u
。可以解释为先用指针指向第三个同学的名字的首地址*(students_name+2)
,然后再取得地址偏移量为4的地址解除引用获取第5个字符*(*(students_name+2)+4)
。
3、但是注意,用*(*(ps+2)+4)
编译报错,*
操作符解除引用操作的是指针(或者说一个地址),此时*(ps+2)+4
不是一个指针,就不能用*
解除引用。而且,用*(ps+2)+4
也是不行的,不知道打印到哪个内存的值了。
4、可以用*(ps+20*2+4)
,printf("%c",*(ps+20*2+4));
将打印出u
。这就是一个字符一个字符的地址加上来,因为每个学生的名字占20个字符,所以第三个学生的第5个字符位置就是20*2+4=44。
编译器在编译时,不管是写的哪种形式获取ch_35,其实都是会先解析成
*(*(students_name+2)+4);
- 总结:对数组char a[10];
1、用a[i]
这样的形式对数组进行访问时,总是先被编译器“改写”或解释为像*(a+1)
这样的指针访问。
2、数组作为函数的形参时,限定数组的长度时,三种方式:void show(char arr[])
或void show(char arr[10])
或void show(char *pa)
都可行,多维数组a[10][20]只能是void show(char arr[][20])
或void show(char (*pa)[20])——【char (*pa)[20]表示pa是一个指向长度为20的字符数组的指针(简称数组指针),而char *pa[20]表示pa是一个有20个元素的数组,数组的每个元素都是一个指向char类型的指针(简称指针数组)】
。
3、多维数组不限定除第一维之外的其他维的长度(可接受任意长度的数组),另外增加1个int型的形参来表示数组的个数,如void show(char **pa, int strcount)——【char **pa; pa是指针的指针,能接受的实参的类型也只能是指针的指针char **pb或者指针数组char *pb[10]】
或者void show(char *pa[], int strcount)
。但是函数内部都是解释成指针形式。
4、注意:数组形参地址&arr或者&pa会和&arr[0]==&pa[0]==&a==&a[0]不一样。因为arr和pa也有自己的定义和分配内存。
5、作为函数调用的实参的数组,始终会被编译器修改成指向数组第一个元素的指针。