关于数组指针

问题描述

如何从此数组指针中获取“ l”?

char *p[3][2] = {"abc","defg","hi","jklmno","pqrstuvw","xyz"};

喜欢

printf("%c\n",*((*p)) + 2);

解决方法

让我们使用大括号括起来的声明使您的数组初始化程序更清晰

char *p[3][2] = {{"abc","defg"},{"hi","jklmno"},{"pqrstuvw","xyz"}};

现在您可以看到p[1]是数组{"hi","jklmno"}
那么您想获取第二个字符串,即p[1][1],然后是第三个字符:p[1][1][2]

这是一个正在运行的示例:https://ideone.com/Jd9GWI

,

如果我已正确理解,您将使用指针从字符串文字"jklmno"中输出字母'l'。

此文字在索引为1的行中,在索引为1的列中。

因此,要使用下标运算符访问文字,您可以编写表达式p[1][1],该表达式产生指向字符串文字第一个字符的指针。现在,要获取字符串文字的字符“ l”,您可以编写表达式p[1][1][2]

printf( "%c\n",p[1][1][2] );

现在让我们使用不带下标运算符的指针来做同样的事情。

在表达式中使用的具有极少数例外的数组指示符将转换为指向其第一个元素的指针。

因此在表达式中使用的数组指示符p将转换为类型为char * ( * )[2]的指针,该指针指向数组的第一行。

要获取指向数组第二行的指针,应编写p + 1

要获取行本身,必须取消引用指针

*( p + 1 )

此表达式的类型为char *[2]

但是由于该表达式是一维数组的代号,然后在表达式中使用,因此它被转换为指向其第一个元素的指针,并且类型为char **

要访问指向该行第二个元素的指针,可以使用指针算法

*( p + 1 ) + 1

要获取指向的元素(字符串文字),必须取消对指针的引用

*( *( p + 1 ) + 1 )

此表达式的类型为char *,并指向字符串文字"jklmno"的第一个字符。

要获取指向字符串文字第三个字符的指针,您必须使用此表达式

*( *( p + 1 ) + 1 ) + 2

现在,您需要取消指针的引用,以输出指向字母的'l'

*( *( *( p + 1 ) + 1 ) + 2 )

所以函数调用看起来像

printf( "%c\n",*( *( *( p + 1 ) + 1 ) + 2 ) );

与先前显示的函数调用进行比较

printf( "%c\n",p[1][1][2] );

请注意,如果您有类似的数组

T a[N];

其中T是某种类型,然后使用下标运算符访问数组的i-th元素,您可以编写表达式a[i]。在幕后,使用诸如*( a + i )之类的指针算法来计算该表达式。由于加法运算符的可交换性质,该表达式也可以像*( i + a )一样编写。因此,使用下标运算符可以编写与表达式i[a]等效的表达式a[i]

根据C标准(6.5.2.1数组下标)

2后缀表达式,后跟方括号[] 是数组对象元素的下标名称。的 下标运算符[]的定义是E1 [E2]与 (*((E1)+(E2)))。由于适用于 二进制+运算符,如果E1是一个数组对象(相当于一个指针) 到数组对象的初始元素),并且E2是整数, E1 [E2]表示E1的第E2个元素(从零开始计数)。

因此表达式

*( *( *( p + 1 ) + 1 ) + 2 )

可能被重写为

*( 2 + *( 1 + *( 1 + p ) ) )

并相应地喜欢

2[1[1[p]]]

和函数printf

的调用
printf( "%c\n",p[1][1][2] );

也可能被重写为(尽管您不应该这样做使代码变得混乱)

printf( "%c\n",2[1[1[p]]] );

这是一个演示程序。

#include <stdio.h>

int main() 
{
    char *p[3][2] = 
    {
        { "abc","defg" },{ "hi","jklmno" },{ "pqrstuvw","xyz" }
    };
    
    printf( "%c\n",2[1[1[p]]] );
}   

其输出为

l