二七十四 发表于 2019-3-21 14:20:03

c语言结构体求助

若某结构型指针变量p已经指向该结构型某数组,错误地引用该结构型数组元素成员的是()
A (*p).成员名          B *(p+i).成员名
C (p+i)->成员名      D p->成员名

如题,书上给的答案是c,但是这本书的答案错误率不低,我不太信任,而且我看不出c错在哪里了。我个人倾向性于B,我认为成员运算符的优先级高于指针运算符,B选项等于是*((p+i).成员名),而(p+i).成员名是错误格式。

我在站内搜到了一个14年的求助贴,里面有一个和这个一样的题,有人说选c,有人说没有选项,全部都对,但是都没有给出理由。

希望有大佬帮我稍微详细解答一下。

在此谢过。

sunrise085 发表于 2019-3-21 15:03:08

本帖最后由 sunrise085 于 2019-3-21 15:36 编辑

你的理解是对的。指针对结构体成员的引用有两种型式,一种是“.”,一种是“->”。这两种的区别就是前者的左边需要是实体,后者的左边需要是指针。那么指针怎么分为实体和指针呢?在使用指针的时候前面加*则成为实体,不加*则是指针,所以这道题的选项C和D都是第二种情况,是对的。然后看选项A和B,选项A中,*p放在了括号内,之后再引用其成员写法是对的,选项B错就错在运算符的优先级问题上了,应该是(*(p+i)).成员名,这样就和选项A一样了。
这个题目写一个简单想小程序就可以测试出来那个是错误选项了。
#include <stdio.h>
struct stu{
    char str;
};
int main(void) {
    stu *p,mystu;
    for(int i=0;i<2;i++){
      mystu.str='a'+i;
    }
    p=mystu;
        printf("str=%c\n",(*p).str);
        printf("str=%c\n",*(p+1).str);
        printf("str=%c\n",(p+1)->str);
        printf("str=%c\n",p->str);
        return 0;
}

风扫地 发表于 2019-3-21 15:06:38

typedef struct tag_test
{
        int a;
        int b;
}test_t;


test_t test_s;
test_t *p= test_s;


test_s.a = 1;
test_s.b = 2;
test_s.a = 3;
test_s.b = 4;
test_s.a = 5;
test_s.b = 6;
test_s.a = 7;
test_s.b = 8;
test_s.a = 9;
test_s.b = 10;

//A (*p).成员名          B *(p+i).成员名
//C (p+i)->成员名      D p->成员名

printf("%d\n",(*p).a)//选项A
printf("%d\n",*(p+1).a   ) //选项B
printf("%d\n",(p+1)->a) //选项C
printf("%d\n",p->a) //选项D

貌似B的写法编译就会报错,应该是    (*(p+1)).a



jackz007 发表于 2019-3-21 15:48:07

本帖最后由 jackz007 于 2019-3-21 15:49 编辑

      错误的是 B,理由是 . 运算符的优先级高于 *,所以,* (p + i) . <成员> 实际上等效于,* ((p + i) . <成员>),p 是一个指针,(p + i) . <成员> 这种表达无疑就是错误的,前面在加上 * 就错得更加没谱了。

二七十四 发表于 2019-3-21 15:58:57

sunrise085 发表于 2019-3-21 15:03
你的理解是对的。指针对结构体成员的引用有两种型式,一种是“.”,一种是“->”。这两种的区别就是前者的 ...

谢谢,你讲的很简单易懂。

二七十四 发表于 2019-3-21 15:59:59

风扫地 发表于 2019-3-21 15:06


谢谢,你的程序思路让我明白,原来还可以这样验证。

二七十四 发表于 2019-3-21 16:00:32

jackz007 发表于 2019-3-21 15:48
错误的是 B,理由是 . 运算符的优先级高于 *,所以,* (p + i) .实际上等效于,* ((p + i) . ),p...

谢谢你的解惑
页: [1]
查看完整版本: c语言结构体求助