C++结构体是自定义数据类型,用struct定义,成员默认public,可包含数据和函数,支持初始化、成员访问及内存对齐,与class主要区别在于默认访问权限。

结构体(
struct
要定义和使用C++结构体,核心就是
struct
1. 结构体的定义
定义结构体,你得先用
struct
{}立即学习“C++免费学习笔记(深入)”;
// 定义一个表示二维点的结构体
struct Point {
int x; // x坐标
int y; // y坐标
};
// 定义一个表示学生信息的结构体
struct Student {
std::string name; // 姓名
int age; // 年龄
double score; // 成绩
};这里需要注意,结构体定义完,别忘了最后那个分号
;
2. 结构体变量的声明与初始化
定义好结构体类型后,你就可以像使用
int
char
// 声明一个Point类型的变量p1 Point p1; // 声明一个Student类型的变量s1 Student s1;
声明变量后,你可以对它们的成员进行赋值。
p1.x = 10; p1.y = 20; s1.name = "张三"; s1.age = 18; s1.score = 95.5;
C++也支持在声明时进行初始化,这叫聚合初始化(aggregate initialization),尤其对于简单的结构体很方便:
Point p2 = {30, 40}; // 按照成员声明的顺序进行初始化
Student s2 = {"李四", 20, 88.0}; // 同样按顺序如果成员是复杂的类型,比如另一个结构体,也可以嵌套初始化:
struct Line {
Point start;
Point end;
};
Line l1 = {{1, 2}, {5, 6}}; // 嵌套初始化3. 结构体成员的访问
访问结构体变量的成员,使用“点操作符”(
.
std::cout << "p1的x坐标是: " << p1.x << std::endl; std::cout << "s1的姓名是: " << s1.name << std::endl;
当你有结构体指针时,你需要使用“箭头操作符”(
->
Point* ptr_p1 = &p1; // 获取p1的地址,赋值给指针 std::cout << "通过指针访问,p1的y坐标是: " << ptr_p1->y << std::endl; // 等价于 (*ptr_p1).y,但箭头操作符更简洁
说实话,C++里的
struct
class
它们最主要的、也是唯一的语法层面的区别,在于默认的成员访问权限。
struct
public
private:
protected:
class
private
public:
protected:
从历史角度看,
struct
class
struct
struct
class
我个人在实践中,如果只是想简单地把一些数据打包,且这些数据对外是透明的,我更倾向于用
struct
class
很多人可能觉得
struct
struct
class
1. 成员函数
在结构体内部定义成员函数,就和在类里定义成员函数一样。这些函数可以操作结构体内部的数据成员。
struct Point {
int x;
int y;
// 成员函数:打印点坐标
void print() {
std::cout << "(" << x << ", " << y << ")" << std::endl;
}
// 成员函数:移动点
void move(int dx, int dy) {
x += dx;
y += dy;
}
};
// 使用示例
Point p;
p.x = 1;
p.y = 2;
p.print(); // 输出 (1, 2)
p.move(5, -3);
p.print(); // 输出 (6, -1)2. 构造函数
构造函数是一种特殊的成员函数,它在创建结构体对象时自动调用,用于初始化对象的状态。构造函数的名称必须和结构体名称相同,并且没有返回类型。
struct Point {
int x;
int y;
// 默认构造函数:不带参数
Point() {
x = 0;
y = 0;
std::cout << "默认构造函数被调用" << std::endl;
}
// 带参数的构造函数:初始化x和y
Point(int initial_x, int initial_y) {
x = initial_x;
y = initial_y;
std::cout << "带参数构造函数被调用" << std::endl;
}
void print() {
std::cout << "(" << x << ", " << y << ")" << std::endl;
}
};
// 使用示例
Point p1; // 调用默认构造函数,p1为(0,0)
p1.print();
Point p2(10, 20); // 调用带参数构造函数,p2为(10,20)
p2.print();通过构造函数,你可以确保结构体对象在创建时就处于一个有效的、有意义的状态,避免了手动逐个初始化成员的麻烦和潜在错误。
这块内容有点技术深度,但对于理解结构体的底层工作原理,以及避免一些潜在的性能问题或跨平台兼容性问题,是相当重要的。C++标准并没有强制规定结构体成员在内存中必须是紧密排列的,编译器在处理结构体时,为了优化内存访问效率,往往会引入“内存对齐”(Memory Alignment)。
1. 内存对齐(Padding)
简单来说,内存对齐就是编译器在结构体成员之间插入一些空白字节(padding),以确保每个成员都从一个内存地址开始,而这个地址是其自身大小(或其类型对齐要求)的整数倍。这样做的好处是,CPU访问对齐的数据通常会更快,因为很多硬件设计要求数据从特定的地址边界开始读取。
举个例子:
struct MyStruct {
char c1; // 1 byte
int i; // 4 bytes
char c2; // 1 byte
};理想情况下,你可能觉得这个结构体的大小是 1 + 4 + 1 = 6 字节。但实际上,它很可能不是6字节。假设在一个32位或64位系统上,
int
c1
i
c1
i
i
c2
此时,结构体的大小看起来是 1 (c1) + 3 (padding) + 4 (i) + 1 (c2) = 9 字节。但通常,结构体的总大小也会被对齐到其最大成员的对齐边界。在这个例子中,
int
c2
你可以使用
sizeof
std::cout << "MyStruct的大小是: " << sizeof(MyStruct) << " 字节" << std::endl; // 输出通常会是 12
2. 为什么内存对齐很重要?
3. 控制对齐
在某些特殊情况下,你可能需要精确控制结构体的对齐方式,例如为了节省内存或者与外部二进制数据格式匹配。C++11引入了
alignas
#pragma pack
// 使用alignas强制对齐
struct alignas(8) AlignedStruct { // 整个结构体8字节对齐
int i;
char c;
};
// 使用#pragma pack (非标准,慎用)
#pragma pack(push, 1) // 设置对齐字节为1,即不进行填充
struct PackedStruct {
char c1;
int i;
char c2;
};
#pragma pack(pop) // 恢复默认对齐
std::cout << "AlignedStruct的大小是: " << sizeof(AlignedStruct) << " 字节" << std::endl; // 可能是8或16
std::cout << "PackedStruct的大小是: " << sizeof(PackedStruct) << " 字节" << std::endl; // 应该是6但我个人经验是,除非你真的清楚自己在做什么,并且有非常明确的性能或兼容性需求,否则最好不要轻易去修改默认的对齐规则。让编译器自己处理通常是最安全和高效的做法。过度干预可能导致代码的可移植性变差,甚至引入难以调试的bug。
以上就是C++结构体如何定义和使用 struct关键字基本语法解析的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号