C—结构体

简介: C语言学习第十四弹

学习目标:


掌握 C 入门知识

学习内容:


一. 结构体类型的声明和初始化


结构是一些值的集合,这些值称为成员变量。结构的每个成员可以是不同类型

(1)声明


#include<stdio.h>
struct Book
{
  char name[20];//成员变量
  int price;
}b3,b4,b5;//全局变量
int main()
{
  struct Book b1;//
  struct Book b2;//局部变量
  return 0;
}

注意上面的写法中b3 b4 b5是全局变量 b1 b2是局部变量;下面写法中Book是类型

C语言允许用户使用 typedef 关键字来定义自己习惯的数据类型名称,来替代系统默认的基本类型名称、数组类型名称、指针类型名称与用户自定义的结构型名称、共用型名称、枚举型名称等。一旦用户在程序中定义了自己的数据类型名称,就可以在该程序中用自己的数据类型名称来定义变量的类型、数组的类型、指针变量的类型与函数的类型等。

#include<stdio.h>
typedef struct Book
{
  char name[20];//成员变量
  int price;
}Book;//类型
int main()
{
  struct Book b1;
  struct Book b2;//局部变量
  Book b3;
  return 0;
}
//当然也可以分两步写
#include<stdio.h>
struct Book
{
  char name[20];//成员变量
  int price;
};
typedef struct Book Book;
int main()
{
  struct Book b1;
  struct Book b2;//局部变量
  Book b3;
  return 0;
}

(2)初始化


#include<stdio.h>
struct S
{
  char c;
  char arr[10];
  int* p;
}s1,s2;
struct S s3;
struct B
{
  int a;
  struct S s;
  double d;
}sb1,sb2;
int main()
{
  struct S s4 = {'c',"张三",NULL};
  int a = 10;
  struct B sb4 = { 100,{'q',"李四",&a},3.14 };
  return 0;
}
#include<stdio.h>
struct Book
{
  char name[20];
  int price;
};
int main()
{
  struct Book s1 = { "你好",100 };
  struct Book s2 = { .price = 68,.name = "再见" };//当然也可以不按照顺序来写,则需要写成.name .price然后赋值
  printf("%s %d\n", s1.name, s1.price);
  printf("%d %s\n", s2.price, s2.name);
  return 0;
}

二.结构体成员的访问


(1)结构体变量访问成员


结构变量的成员是通过点操作符(.)访问的。点操作符接受两个操作数。

#include<stdio.h>
struct Book
{
  char name[20];
  int price;
};
int main()
{
  struct Book s1 = { "你好",100 };
  struct Book s2 = { .price = 68,.name = "再见" };//当然也可以不按照顺序来写,则需要写成.name .price然后赋值
  printf("%s %d\n", s1.name, s1.price);
  printf("%d %s\n", s2.price, s2.name);
  return 0;
}

(2)结构体指针访问指向变量的成员


有时候我们得到的不是一个结构体变量,而是指向一个结构体的指针。

#include<stdio.h>
struct Book
{
  char name[20];
  int price;
};
void print(struct Book* p)
{
  printf("%s %d\n", p->name, p->price);
}
int main()
{
  struct Book s1 = { "你好",100 };
  struct Book s2 = { .price = 68,.name = "再见" };
  //结构体指针->结构体成员
  print(&s1);
  return 0;
}

三.结构体传参


第一种:


#include<stdio.h>
struct S
{
  int arr[100];
  int n;
};
void print(struct S ss)//ss是形参
{
  int i = 0;
  for (i = 0; i < 10; i++)
  {
    printf("%d ", ss.arr[i]);
  }
  printf("\n");
  printf("%d\n", ss.n);
}
int main()
{
  struct S s = { {1,2,3,4,5},100 };
  print(s);//s是实参
  return 0;
}

第二种:


#include<stdio.h>
struct S
{
  int arr[100];
  int n;
};
void print1(struct S* ps)
{
  int i = 0;
  for (i = 0; i < 10; i++)
  {
    printf("%d ", ps->arr[i]);
  }
  printf("\n");
  printf("%d\n", ps->n);
}
int main()
{
  struct S s = { {1,2,3,4,5},100 };
  print1(&s);
  return 0;
}

以上两种传参哪种更好呢?

答案:第二种

原因:函数传参的时候,参数是需要压栈的。

如果传递一个结构体对象的时候,结构体过大,参数压栈的的系统开销比较大,所以会导致性能的下降。

结论😗


结构体传参的时候,要传结构体的地址。


相关文章
|
10月前
|
10月前
|
缓存 NoSQL 关系型数据库
大厂面试高频:如何解决Redis缓存雪崩、缓存穿透、缓存并发等5大难题
本文详解缓存雪崩、缓存穿透、缓存并发及缓存预热等问题,提供高可用解决方案,帮助你在大厂面试和实际工作中应对这些常见并发场景。关注【mikechen的互联网架构】,10年+BAT架构经验倾囊相授。
大厂面试高频:如何解决Redis缓存雪崩、缓存穿透、缓存并发等5大难题
|
缓存 安全 Java
原子操作的实现原理
原子操作的实现原理
234 0
|
开发者 Python
基于Python的日志管理与最佳实践
日志是开发和调试过程中的重要工具,然而,如何高效地管理和利用日志常常被忽略。本文通过Python中的logging模块,探讨如何使用日志来进行调试、分析与问题排查,并提出了一些实际应用中的优化建议和最佳实践。
|
机器学习/深度学习 数据采集 存储
一文读懂蒙特卡洛算法:从概率模拟到机器学习模型优化的全方位解析
蒙特卡洛方法起源于1945年科学家斯坦尼斯劳·乌拉姆对纸牌游戏中概率问题的思考,与约翰·冯·诺依曼共同奠定了该方法的理论基础。该方法通过模拟大量随机场景来近似复杂问题的解,因命名灵感源自蒙特卡洛赌场。如今,蒙特卡洛方法广泛应用于机器学习领域,尤其在超参数调优、贝叶斯滤波等方面表现出色。通过随机采样超参数空间,蒙特卡洛方法能够高效地找到优质组合,适用于处理高维度、非线性问题。本文通过实例展示了蒙特卡洛方法在估算圆周率π和优化机器学习模型中的应用,并对比了其与网格搜索方法的性能。
1018 1
|
JavaScript
vue组件封装 —— 仪表盘(有缺口的环形进度条内显示百分比值)
vue组件封装 —— 仪表盘(有缺口的环形进度条内显示百分比值)
406 1
|
数据库 数据中心 虚拟化
云服务器和虚拟主机有什么区别
总的来说,云服务器提供更高级别的灵活性、控制和资源隔离,适用于需要定制化和更大资源的应用程序。虚拟主机是一种经济实惠的共享托管解决方案,适用于小型项目和个人网站。选择哪种服务取决于你的需求、预算和技术要求。
299 3
|
监控 项目管理 调度
深入探究ERP系统的项目管理模块
深入探究ERP系统的项目管理模块
291 3
|
JavaScript 前端开发 Java
JavaScript高级笔记-coderwhy版本(六)
JavaScript高级笔记-coderwhy版本
243 0
|
存储 关系型数据库 MySQL
分库分表:存量1亿,日增量500万如何分库分表?
分库分表:存量1亿,日增量500万如何分库分表?
277 0