http://c.biancheng.net/view/246.html
struct STUDENT
{
char name[20]; //姓名
int num; //学号
struct AGE // 生日
{
int year;
int month;
int day;
};
float score; //分数
}student1;
1、&student1 表示结构体变量 student1 的首地址,即 student1 第一个项的地址,那么如果定义一个指针变量 p 指向这个地址的话,p 就可以指向结构体变量 student1 中的任意一个成员了
2、我们知道,指针的声明其类型一定要与指向的目标类型一致(可以理解:方便使用*运算的时候让机器明确读取的数据类型),所以一般我们如此定义一个结构体指针:
struct STUDENT *p = NULL;
p = & student1;
3、看看,在程序段中如何使用这样的指针:
int main(void)
{
struct STUDENT student1; /*用struct STUDENT结构体类型定义结构体变量student1*/
struct STUDENT *p = NULL; /*定义一个指向struct STUDENT结构体类型的指针变量p*/
p = &student1; /*p指向结构体变量student1的首地址, 即第一个成员的地址*/
strcpy((*p).name, "小明"); //(*p).name等价于student1.name
(*p).birthday.year = 1989;
(*p).birthday.month = 3;
(*p).birthday.day = 29;
(*p).num = 1207041;
(*p).score = 100;
printf("name : %s\n", (*p).name); //(*p).name不能写成p
printf("birthday : %d-%d-%d\n", (*p).birthday.year, (*p).birthday.month, (*p).birthday.day);
printf("num : %d\n", (*p).num);
printf("score : %.1f\n", (*p).score);
return 0;
}
输出结果是:
name : 小明
birthday : 1989-3-29
num : 1207041
score : 100.0
我们看到,用指针引用结构体变量成员的方式是:
(*指针变量名).成员名
(*p).score
(*p).birthday.year
注意
1、由于点“.”运算的优先级高于星“*”,所以(*p)括号不能省略
2、指针p指向的是结构体变量student1的第一个成员地址,也便是name的首地址,所以地址数值上 p == (*p).name;
3、但是我们都学过数学,也知道纯数字是不具有任何信息属性的,带上他们的数据类型来看,虽然他们的地址相同但是他们的类型确实不同的,所以他们的含义也是本质不一样。
指针变量 p 是 struct STUDENT* 型的,而 (*p).name 是 char* 型的。
同样,虽然 &student1 和 student1.name 表示的是同一个内存单元的地址,但它们的类型是不同的。
&student1 是 struct STUDENT* 型的,而 student1.name 是 char* 型的,所以在对 p 进行初始化时,“p=&student1;”不能写成“p=student1.name”。
因为 p 是 struct STUDENT* 型的,所以不能将 char* 型的 student1.name 赋给 p。
总结拓展:
以下 3 种形式是等价的:
结构体变量.成员名。
(*指针变量).成员名。
指针变量->成员名。
只有“指针变量名”后面才能加“->”,千万不要在成员名如 birthday 后面加“->”。