知识点11:
首先让我们定义结构体:
struct stu
{
char name[20];
long number;
float score[4];
} ;
再定义指向结构体类型变量的指针变量:
struct stu *p1, *p2 ;
定义指针变量p 1、p 2,分别指向结构体类型变量。引用形式为:指针变量→成员;
[例7-2] 对指向结构体类型变量的正确使用。输入一个结构体类型变量的成员,并输出。
#include <stdlib.h> /*使用m a l l o c ( ) 需要* /
struct data / *定义结构体* /
{
int day,month,year;
} ;
struct stu /*定义结构体* /
{
char name[20];
long num;
struct data birthday; /*嵌套的结构体类型成员*/
} ;
main() /*定义m a i n ( ) 函数* /
{
struct stu *student; /*定义结构体类型指针*/
student=malloc(sizeof(struct stu)); /*为指针变量分配安全的地址*/
printf("Input name,number,year,month,day:/n");
scanf("%s",student->name); /*输入学生姓名、学号、出生年月日*/
scanf("%d",&student->num);
scanf("%d%d%d",&student->birthday.year,&student->birthday.month,
&student->birthday.day);
printf("/nOutputname,number,year,month,day/n");/*打印输出各成员项的值*/
printf("%20s%10ld%10d//%d//%d/n",student->name,student->num,
student->birthday.year,student->birthday.month,
student->birthday.day);
}
程序中使用结构体类型指针引用结构体变量的成员,需要通过C提供的函数malloc()来为
指针分配安全的地址。函数sizeof()返回值是计算给定数据类型所占内存的字节数。指针所指
各成员形式为:
student->name
student->num
student->birthday.year
student->birthday.month
student->birthday.day
指向结构体类型数组的指针的使用
定义一个结构体类型数组,其数组名是数组的首地址,这一点前面的课程介绍得很清楚。
定义结构体类型的指针,既可以指向数组的元素,也可以指向数组,在使用时要加以区分。
[例7-3] 在例7 - 2中定义了结构体类型,根据此类型再定义结构体数组及指向结构体类型的指针。
struct data
{
intday,month,year;
};
struct stu/*定义结构体*/
{
char name[20];
long num;
struct data birthday;/嵌*套的结构体类型成员*/
};
struct stu student[4],*p;定/*义结构体数组及指向结构体类型的指针*/
作p=student,此时指针p就指向了结构体数组student。
p是指向一维结构体数组的指针,对数组元素的引用可采用三种方法。
1)地址法
student+i和p+i均表示数组第i个元素的地址,数组元素各成员的引用形式为:
(student+i)->name、(student+i)->num和(p+i)->name、(p+i)->num等。student+i和p+i
与&student[i]意义相同。
2)指针法
若p指向数组的某一个元素,则p++就指向其后续元素。
3)指针的数组表示法
若p=student,我们说指针p指向数组student,p[i]表示数组的第i个元素,其效果与
student[i]等同。对数组成员的引用描述为:p[i].name、p[i].num等。
[例7-4]指向结构体数组的指针变量的使用。
structdata/*定义结构体类型*/
{
intday,month,year;
};
structstu/*定义结构体类型*/
{
char name[20];
long num;
struct data birthday;
};
main()
{inti;
structstu*p,student[4]={{"liying",1,1978,5,23},{"wangping",2,1979,3,14},
{"libo",3,1980,5,6},{"xuyan",4,1980,4,21}};
/*定义结构体数组并初始化*/
p=student;/*将数组的首地址赋值给指针p,p指向了一维数组student*/
printf("/n1----Outputname,number,year,month,day/n");
for(i=0;i<4;i++)/*采用指针法输出数组元素的各成员*/
printf("%20s%10ld%10d//%d//%d/n",(p+i)->name,(p+i)->num,
(p+i)->birthday.year,(p+i)->birthday.month,
(p+i)->birthday.day);
}
针对时间获取函数的理解:函数功能为打印时间
下面是本人编写的三种方式,其中一种方式为错误方式
方式1
在rth.h中
typedef struct
{
uint16 year; //年
uint8 mon; //月
uint8 day; //日
uint8 week; //星期
uint8 hour; //时
uint8 min; //分
uint8 sec; //秒
}TIMER;
void GetTimer(TIMER *ptr);
在rtc.c中定义
void GetTimer(TIMER *p)
{
p->year =YEAR;
p->mon =MONTH;
p->day =DOM ;
p->week =DOW ;
p->hour =HOUR;
p->min =MIN ;
p->sec =SEC ;
}
在打印函数中
void handle_get_rtc(int argc,char *argv[])
{
TIMER time;
GetTimer(&time);
MY_DEBUGF(CMD_LINE_DEBUG,("%d %d %d %d %d %d\n",time.year,time.mon,time.day,time.hour,time.min,time.sec));
MY_DEBUGF(CMD_LINE_DEBUG,("--------------------------------------------------\n"));
}
方式2
在rth.h中
typedef struct
{
uint16 year; //年
uint8 mon; //月
uint8 day; //日
uint8 week; //星期
uint8 hour; //时
uint8 min; //分
uint8 sec; //秒
}TIMER;
void GetTimer(TIMER ptr);
在rtc.c中定义
void GetTimer(TIMER p)
{
p.year =YEAR;
p.mon =MONTH;
p.day =DOM ;
p.week =DOW ;
p.hour =HOUR;
p.min =MIN ;
p.sec =SEC ;
}
在打印函数中
void handle_get_rtc(int argc,char *argv[])
{
TIMER time;
GetTimer(time);
MY_DEBUGF(CMD_LINE_DEBUG,("%d %d %d %d %d %d\n",time.year,time.mon,time.day,time.hour,time.min,time.sec));
MY_DEBUGF(CMD_LINE_DEBUG,("--------------------------------------------------\n"));
}
方式1与方式2究竟谁正确呢?经过测试发现方式1正确,那究竟方式2错在哪了呢?
在方式1中,GetTimer(TIMER *p),定义了已结构体类型的指针p,p是一个地址,再定义TIMER time; time是一个结构体类型的变量,他存放的地址为&time,在调用GetTimer(&time); 函数时相当于将p地址里的数据放到&time地址里面去,传递的是地址,而方式2则传递的是整个结构体,在c中,整个结构体是不能传递的。既然整个结构体不能传递,那么是否可以定义一个全局型的结构体变量呢?答案是可以的,如方式3
方式3
在rth.h中
typedef struct
{
uint16 year; //年
uint8 mon; //月
uint8 day; //日
uint8 week; //星期
uint8 hour; //时
uint8 min; //分
uint8 sec; //秒
}TIMER;
extern TIMER time;
void GetTimer(void);
在rtc.c中定义
TIMER time;
void GetTimer(void)
{
time.year =YEAR;
time.mon =MONTH;
time.day =DOM ;
time.week =DOW ;
time.hour =HOUR;
time.min =MIN ;
time.sec =SEC ;
}
在打印函数中
void handle_get_rtc(int argc,char *argv[])
{
GetTimer();
MY_DEBUGF(CMD_LINE_DEBUG,("%d %d %d %d %d %d\n",time.year,time.mon,time.day,time.hour,time.min,time.sec));
MY_DEBUGF(CMD_LINE_DEBUG,("--------------------------------------------------\n"));
}
方式1与方式3相比,方式1更好,因为采用方式1的话不仅&time可以来获取时间时,我们还可以定义TIMER time1;TIMER time2;等等,方式1更具有通用性