美文网首页C/C++经验技巧总结
C 多维数组元素的指针表示问题

C 多维数组元素的指针表示问题

作者: XDgbh | 来源:发表于2018-03-22 12:36 被阅读3次

    比如要定义(开辟内存)一个字符数组,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、作为函数调用的实参的数组,始终会被编译器修改成指向数组第一个元素的指针。

    相关文章

      网友评论

        本文标题:C 多维数组元素的指针表示问题

        本文链接:https://www.haomeiwen.com/subject/dwtaqftx.html