/**********************************************************************
* 版权所有 (C)2017, Wang maochun。
*
* 文件名称:stu.c
* 文件标识:无
* 内容摘要:演示程序
* 其它说明:主要学习三点知识:
1.建立一个结构体,类型名为student
2.当定义结构体变量时,通过成员操作符访问成员(.)
当定义指向结构体的指针时,通过间接访问操作符(->)
3.传值操作,子函数对实参进行了一份拷贝,对其的操作不会影响到orginal value,我们通过打印地址和实参不同可以看到
传址操作,子函数传递了实参地址,此时操作的是orginal value,优点是节省空间和花销。如果子函数不修改需const修饰
* 当前版本:V1.0
* 作 者:Wang maochun
* 完成日期:2017.7.23
*
**********************************************************************/ #include <stdio.h>
struct student{
int code;
char name[];
char sex[];
int age;
};
void printStuInfobyValue(student stu)
{
printf("in fun结构体传值:学号、姓名、性别、年龄:\n");
printf("%5d [%p]\n %5s [%p]\n %5s [%p]\n %5d [%p]\n\n",stu.code,&stu.code,stu.name,&stu.name,
stu.sex,&stu.sex,stu.age,&stu.age);
} void printStuInfoByAdress(student* pstu)
{
printf("in fun结构体传地址:学号、姓名、性别、年龄:\n");
printf("%6d [%p]\n %6s [%p]\n %6s [%p]\n %6d [%p]\n\n",pstu->code,&pstu->code,pstu->name,&pstu->name,
pstu->sex,&pstu->sex,pstu->age,&pstu->age);
} int main(){
//struct student stu;
student stu = {,"wsq","Male",};
student *pstu = &stu;
printf("in main:学号、姓名、性别、年龄:\n");
//scanf("%d %s %s %d",&stu.code,stu.name,stu.sex,&stu.age);
printf("%5d [%p]\n %5s [%p]\n %5s [%p]\n %5d [%p]\n\n",stu.code,&stu.code,stu.name,&stu.name,
stu.sex,&stu.sex,stu.age,&stu.age);
printf("%6d [%p]\n %6s [%p]\n %6s [%p]\n %6d [%p]\n\n",pstu->code,&pstu->code,pstu->name,&pstu->name,
pstu->sex,&pstu->sex,pstu->age,&pstu->age); printStuInfobyValue(stu); //结构体传值
printStuInfoByAdress(&stu); //结构体传地址
return ;
}
运行效果:
我们可以看到:
结构体对象作为参数时,编译器对其进行了copy,(我们通过传入的地址和main中不同可以发现)。此时在函数中的操作都是对其拷贝的操作,不影响main函数中的origin value
缺点是,当结构体变量非常大时,编译器对其进行复制,开销较大。
用结构体变量作实参时,采取的是"值传递"的方式,将结构体变量所占的内存单元的内容全部顺序传递给形参.形参也必须是同类型的结构体变量.在函数调用期间形参也要占用内存单元. 这种传递方式在空间和时间上开销较大,如果结构体的规模很大时,开销是很可观的.
结构体地址作为参数时,子函数中操作和main函数操作的是同一个结构体,此时传递的参数时一个地址。
优点是不需要进行copy,但是使用时要小心,如果不想修改其值,需用const关键字修饰