C++ struct 笔记(超级详细)

news2025/1/23 13:11:29

今日碎碎念:我在学C语言时经常用到结构体struct,之后在写C++程序时遇到在struct中定义构造函数和成员函数的情况,这在c语言中是从未遇到过的,觉得奇怪,想到之前并没有真正系统学习C++里的struct,有必要今天详细记录一下。

   虽然今天结构体较少使用,但知道它们是什么,以及如何使用它们仍然很重要,这并不仅仅是因为可以在较老的程序中遇到它们,还因为在某些情况(这里请看二、C++ class和struct的区别)下,类的实例无法使用,这时必须使用结构体。

目录

一、C++ struct的用法

    1、struct是什么

     2、 struct的4种声明定义方式

     3、结构体的2种初始化方式

     4、结构体成员的访问

     5、结构体的嵌套

     6、将结构体作为参数和返回值

二、C++ class和struct的区别

三、C++ struct和 c struct的区别 

四、struct计算结构体某个成员相对于结构体基址的偏移


一、C++ struct的用法

    1、struct是什么

        struct是程序员定义的数据类型,将逻辑上连接在一起的不同类型的数据组合到一起的单元。

     2、 struct的4种声明定义方式

           a. 第一种语法表示

        struct 结构体名称
        {
           数据类型 member1;
           数据类型 member2;
        }; 

#include<iostream>
#include <string>
using namespace std;
struct SStudent
{
    int nNo;
    std::string  strName;
};

int main(int argc, char *argv[])
{
    //使用初始化列表初始化struct
    struct SStudent s1 = { 1, "ying"}; //C++定义struct变量时,前面的struct可以写
    SStudent s2 = { 2, "Ming"};  //C++定义struct变量时,前面的struct也可以不写
    cout << s1.nNo << endl;
    cout << s1.strName << endl;
    cout << s2.nNo << endl;
    cout << s2.strName << endl;
    return 0;
}

        b.第二种语法表示

        typedef struct 结构体名称{
           数据类型 member1;
           数据类型 member2;
        }结构体名称别名;
       

  这种情况使用typedef关键字,声明了数据类型的别名,所以在定义结构体 变量时有两种方式:
    第一种:结构体名称 构体变量名
    第二种:结构体名称别名 结构体变量名

#include<iostream>
#include <string>
using namespace std;

typedef struct SStudent
{
    int nNo;
    std::string  strName;
} SStud;

int main(int argc, char *argv[])
{
    SStudent s1 = { 1, "ying"}; //使用结构体名称
    SStud s2 = { 2, "Ming"};    //使用结构体名称别名
    cout << s1.nNo << endl;
    cout << s1.strName << endl;
    cout << s2.nNo << endl;
    cout << s2.strName << endl;
    return 0;
}

   c.第三种语法表示

        struct 结构体名称{
           数据类型 member1;
           数据类型 member2;
        }结构体变量;

#include<iostream>
#include <string>
using namespace std;
struct SStudent
{
    int nNo;
    std::string  strName;
} stu;

int main(int argc, char *argv[])
{
//    stu s2; //错误,没有使用typedef关键字,stu是变量不是类型
    stu = { 1, "ying"}; //使用初始化表赋值
    stu.nNo = 2;        //修改成员nNo值
    stu.strName = "XiaoMing";//修改成员strName值
    cout << stu.nNo << endl;
    cout << stu.strName << endl;
    return 0;
}

   d.第四种语法表示

   //匿名结构体

        struct {
           数据类型 member1;
           数据类型 member2;
        }结构体变量名;

在声明的同时定义了结构体变量,但不能在其它地方声明,因为我们无法得知该结构体的标识符,所以就无法通过标识符来声明变量。

#include<iostream>
#include <string>
using namespace std;

//声明了结构体并定义(且只能在这里定义)了两个该结构体的变量,
//由于无法获取结构体名称(因为匿名),所以无法在其他地方定义该结构体变量
struct
{
    int nNo;
    std::string  strName;
} stu1, stu2;

int main(int argc, char *argv[])
{ 
    stu1 = { 1, "ying"}; //使用初始化表赋值
    stu2 = { 2, "Ming"}; //使用初始化表赋值
    cout << stu1.nNo << endl;
    cout << stu1.strName << endl;
    cout << stu2.nNo << endl;
    cout << stu2.strName << endl;
    return 0;
}

3、结构体的2种初始化方式

   a. 使用初始化列表

     语法:

         //初始化列表中的项目是按照结构体声明中成员的顺序依次赋值,各个成员的赋值用逗号分隔并用大括号括起来

        结构体类型 变量名 = {member1取值,member2取值,member3取值...};

         前面的例子都使用了初始化列表来初始化,这里不再记录例子。

        几个注意点:

        1)使用初始化列表时可以仅仅初始化部分成员,如果某个成员未被初始化,则所有跟在它后面的成员都需要保留为未初始化。C++未提供跳过某个成员初始化其他成员的方法。

#include<iostream>
#include <string>
using namespace std;

//声明了结构体,增加nAge成员,并且定义了两个变量stu1和stu2
struct SStudent
{
    int nNo;
    std::string  strName;
    int nAge;
} stu1, stu2;

int main(int argc, char *argv[])
{ 
    stu1 = { 1, "ying"}; //合法,仅初始化了nNo和strName,不初始化nAge
//    stu2 = { 2, 11}; //非法,不能跳过strName给nAge赋值
    cout << stu1.nNo << endl;
    cout << stu1.strName << endl;
    cout << stu1.nAge << endl;
    cout << stu2.nNo << endl;
    cout << stu2.strName << endl;
    cout << stu2.nAge << endl;
    return 0;
}

       2)可以在结构体声明中使用默认值初始化结构体成员。

#include<iostream>
#include <string>
using namespace std;

//声明了结构体,并且定义了两个变量stu1和stu2
struct SStudent
{
    int nNo = 1;
    std::string  strName = "YunCai";
    int nAge = 6;
} stu1, stu2;

int main(int argc, char *argv[])
{ 
    stu1 = { 2, "ying"}; //不给nAge赋值,将取值默认值6
    cout << stu1.nNo << endl;
    cout << stu1.strName << endl;
    cout << stu1.nAge << endl;
    cout << stu2.nNo << endl;    //不给stu2赋值,所有成员都使用默认值
    cout << stu2.strName << endl;
    cout << stu2.nAge << endl;

    return 0;
}

   运行结果:

   b. 使用构造函数

        与类构造函数一样,结构体的构造函数必须是与结构体名称相同的公共成员函数,并且没有返回类型。因为默认情况下,所有结构体成员都是公开的,所以不需要使用关键字 public。

#include<iostream>
#include <string>
using namespace std;

//声明了结构体,
//并且定义了一个带有三个默认参数的构造函数
//在定义SStudent 变量而不向其传递任何参数时,提供默认值
struct SStudent
{
    int m_nNo;
    string  m_strName;
    int m_nAge;
    SStudent(int nNo = 1, string strName = "YunCai", int nAge = 6)
    {
        m_nNo = nNo;
        m_strName = strName;
        m_nAge = nAge;
    }
};

int main(int argc, char *argv[])
{ 
    SStudent stu1; //调用构造函数并且使用默认参数
    SStudent stu2(2, "ying"); //调用构造函数并传参nNo = 2, strName = "ying"
    cout << stu1.m_nNo << endl;
    cout << stu1.m_strName << endl;
    cout << stu1.m_nAge << endl;
    cout << stu2.m_nNo << endl;
    cout << stu2.m_strName << endl;
    cout << stu2.m_nAge << endl;
    return 0;
}

4、结构体成员的访问

结构体数据成员都是public的,所以它们可以被直接访问,并且可以像常规变量一样使用。

结构体变量使用点运算符.访问数据成员,结构体指针使用->指针运算符访问数据成员。

#include<iostream>
#include <string>
using namespace std;

struct SStudent
{
    int m_nNo;
    string  m_strName;
    int m_nAge;
    SStudent(int nNo = 1, string strName = "YunCai", int nAge = 6)
    {
        m_nNo = nNo;
        m_strName = strName;
        m_nAge = nAge;
    }
};

int main(int argc, char *argv[])
{ 
    SStudent stu1; //调用构造函数并且使用默认参数
    SStudent stu2(2, "ying"); //调用构造函数并传参nNo = 2, strName = "ying"
    SStudent* stu3 = new SStudent(3, "Ming", 8); //使用new初始化SStudent对象指针
    cout << stu1.m_nNo << endl;//使用点运算符访问
    cout << stu1.m_strName << endl;
    cout << stu1.m_nAge << endl;
    cout << stu2.m_nNo << endl;
    cout << stu2.m_strName << endl;
    cout << stu2.m_nAge << endl;
    cout << stu3->m_nNo << endl;//使用指针运算符访问
    cout << stu3->m_strName << endl;
    cout << stu3->m_nAge << endl;
    delete stu3;
    stu3 = nullptr;
    return 0;
}

5、结构体的嵌套

就像一个类的对象可以作为成员放在另一个类中,一个结构体的变量可以作为成员放在另一个结构体中。

#include<iostream>
#include <string>
using namespace std;

//声明了结构体SDate日期
struct SDate
{
    int m_nYear;
    int m_nMonth;
    int m_nDay;
    SDate(const int& nY = 2000, const int& nM = 11, const int& nD = 2)
    {
        m_nYear = nY;
        m_nMonth = nM;
        m_nDay = nD;
    }
};

//声明了结构体SStudent学生,其中m_dateBirth是学生出生日期
struct SStudent
{
    int m_nNo;
    string  m_strName;
    int m_nAge;
    SDate m_dateBirth;
    SStudent(const int& nNo = 1, const string& strName = "YunCai",
             const int&  nAge = 6, const SDate& dateBirth= SDate())
    {
        m_nNo = nNo;
        m_strName = strName;
        m_nAge = nAge;
        m_dateBirth = dateBirth;
    }
};

//使用一个函数打印学生信息,参数类型为常量引用SStudent
void print(const SStudent& stu)
{
    cout << stu.m_nNo << " " << stu.m_strName << "  " << stu.m_nAge << "  ";
    cout <<"Birthday:" << stu.m_dateBirth.m_nYear << "_" <<
           stu.m_dateBirth.m_nMonth << "_" << stu.m_dateBirth.m_nDay << endl;
}

//修改学号增加100,参数类型为值传递,返回值为SStudent
SStudent UpdateNo1(SStudent stu)
{
    stu.m_nNo += 100;
    return stu;
}

//修改学号增加100,参数类型为引用传递
void UpdateNo2(SStudent& stu)
{
    stu.m_nNo += 100;
}

int main(int argc, char *argv[])
{ 
    SStudent stu1; //调用构造函数并且使用默认参数
    print(stu1);
    stu1 =  UpdateNo1(stu1);
    print(stu1);
    SDate stu2_birthday(1999, 9, 10);
    SStudent stu2(2, "Ying", 7, stu2_birthday);//调用够用构造函数并且传了4个参数
    print(stu2);
    UpdateNo2(stu2);
    print(stu2);
    return 0;
}

运行结果:

6、将结构体作为参数和返回值

  a.将结构体作为参数:与类对象一样,结构体变量也可以通过值、引用和常量引用传递给函数。

        1)   值传递:需要生成整个原始结构的副本并传递给函数。因为不希望浪费时间来复制整个结构体,所以,除非结构很小,否则一般会通过引用将结构体传递给函数。

        2)引用传递:引用传递不会生成原始结构体的副本,函数可以直接访问原始结构体的成员变量,也可能更改它们。

        3)常量引用传递:常量引用传递也不会生成原始结构体的副本,函数直接访问原始结构体的成员变量但不能修改原始结构体的成员变量。

b.将结构体作为返回值:函数的返回类型是结构体的名称。

在5、结构体的嵌套中的例子,print函数使用了常量引用传递进行传值,UpdateNo1函数的参数类型为值传递,返回值为结构体,UpdateNo2函数的参数类型为引用传递。

二、C++ class和struct的区别

      结构体和类基本雷同,唯一区别是,类中成员变量默认为私有private,而结构体中则为公有public。因此在使用时,我们可以根据不同的场景或者需求来选择使用struct或者class。

         a.这些情况下使用struct比class更好

                1)纯数据结构:如果一个类只包含数据成员,而没有成员函数;

                 2)数据成员全部为public;

                3)用于C接口:如果一个类需要与C语言交互,例如作为C语言库的接口,那么使用struct更加合适(因为C语言不支持类的概念,而使用struct可以更加方便地进行数据传递);

                4)继承自C结构体:如果一个类需要继承自一个C语言的结构体,那么使用struct更加合适。因为C结构体默认为public,并且在C++中可以使用struct来继承。

        b.这些情况下使用class比struct更好

                1)需要保证数据的安全性,对数据的访问控制比较严格:使用class可以将数据成员设置为私有成员private,防止外部直接修改数据;

                2)需要进行多态:如果一个类需要进行多态操作,例如需要使用虚函数,那么使用class更加合适。

三、C++ struct和 c struct的区别 

        a. C++中定义结构体变量时可以省略struct,但C语言中不可以省略;

        b. C++中struct可以和类一样,有访问权限,并可以定义成员函数; C语言中struct没有访问权限的设置,是一些变量的集合体,不能定义成员函数;

        c.C++中struct可以继承,也可以实现多态;C语言中struct不支持继承和多态;

四、struct计算结构体某个成员相对于结构体基址的偏移

如何计算:

           #define offsetof(s,m) (size_t)&(((s *)0)->m)

解释:

         ((s *)0):强制转化成数据结构指针,并使其指向地址0;
        ((s *)0)->m:使该指针指向成员m
        &(((s *)0)->m):获取该成员m的地址
        (size_t)&(((s *)0)->m):转化这个地址为合适的类型

使用场景

        offsetof已经被定义在系统库中,所以直接使用offsetof计算结构体某个成员相对于结构体基址的偏移

例子:

        要计算如下SDate中m_nDay成员相对于结构体基址的偏移,可以使用 offsetof(SDate, m_nDay)。

        struct SDate{
                int m_nYear;
                int m_nMonth;
                int m_nDay;
        };



参考资料如下:

1、C++ struct的4种定义方式_c++ struct 定义_Mr顺的博客-CSDN博客

2、C++结构体完全攻略(超详细) 

3、什么时候以struct 替代class? - 知乎

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/965903.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

7.6 递归求学生年龄

有5个学生坐在一起&#xff0c;问第五个学生多少岁&#xff0c;他说比第四个学生大两岁。问第四个学生的岁数&#xff0c;他说比第三个学生大2岁。问第三个学生&#xff0c;又说比第二个学生大2岁。问第二个学生&#xff0c;说比第一个学生大2岁。最后问第一个学生&#xff0c;…

Unity中Shader的遮罩的实现

文章目录 前言一、遮罩效果的实现主要是使用对应的纹理实现的&#xff0c;在属性中暴露对应的遮罩纹理&#xff0c;对其进行采样后&#xff0c;最后相乘输出即可二、如果需要像和主要纹理一样流动&#xff0c;则需要使用和_Time篇一样的方法实现流动即可 前言 Unity中Shader的…

TBOX开发需求说明

TBOX功能需求&#xff1a; 支持4G上网功能&#xff0c;可获取外网IP&#xff0c;可和云端平台连通支持路由功能&#xff0c;支持计算平台、网关和云端平台建立网络连接支持USB转网口&#xff0c;智能座舱会通过USB连接AG35建立网络连接&#xff08;类似IVI通过USB口连接TBOX&a…

AAC处理码流分析工具(三十六)

简介: CSDN博客专家,专注Android/Linux系统,分享多mic语音方案、音视频、编解码等技术,与大家一起成长! 优质专栏:Audio工程师进阶系列【原创干货持续更新中……】🚀 人生格言: 人生从来没有捷径,只有行动才是治疗恐惧和懒惰的唯一良药. 更多原创,欢迎关注:Android…

安装kali虚拟机镜像的坑

1.0 安装虚拟机镜像成功之后&#xff0c;只有光标&#xff0c;没有界面 在VMware上安装kali linux环境时&#xff0c;根据提示操作完成后&#xff0c;开启虚拟机&#xff0c;屏幕黑屏&#xff0c;左上角有一个光标在闪&#xff0c;一直开不了机。 出现问题的原因&#xff0c;…

Data truncation: Out of range value for column ‘id‘ at row 1

错误信息&#xff1a;Data truncation: Out of range value for column id at row 1 数据截断&#xff1a;第1行“id”列的值超出范围 很多人会回复&#xff1a;数据库 类型由int改为 bigInt 我看了表结构 可以放的下的。 是 bigint(20) 没有问题啊。 默认的 bigint 类型…

FPGA原理与结构——FIFO IP核的使用与测试

一、前言 本文介绍FIFO Generator v13.2 IP核的具体使用与例化&#xff0c;在学习一个IP核的使用之前&#xff0c;首先需要对于IP核的具体参数和原理有一个基本的了解&#xff0c;具体可以参考&#xff1a; FPGA原理与结构——FIFO IP核原理学习https://blog.csdn.net/apple_5…

GEE14:提取每年的GPP最大值

获取GPP最大值 1. 数据介绍2. JavaScript代码 最近学习了关于获取每年GPP最大值的DOY&#xff08;day of year&#xff09;的方法&#xff1a; 1. 数据介绍 MOD17A2H v006&#xff1a; The MOD17A2H Version 6 Gross Primary Productivity (GPP) product is a cumulative 8-d…

soundtouch库的编译与使用

源码下载 https://gitlab.com/soundtouch/soundtouch/-/archive/2.1.2/soundtouch-2.1.2.tar.bz2 SDK配置 使用vs逐个打开source下指定的三个项目文件&#xff0c;修改SDK&#xff0c;因为可能库中使用的是8.0&#xff0c;你使用的10.0 编译 打开vs&#xff0c;打开终端&am…

图像处理简介

目录 基本术语 1 .图像(image) 1.1 像素(Pixel) 1.2 颜色深度&#xff08;Color Depth&#xff09; 1.3 分辨率&#xff08;Resolution&#xff09; 1.4 像素宽高比&#xff08;Pixel Aspect Ratio&#xff09; 1.5 帧率(FPS) 1.6 码率&#xff08;BR&#xff09; 1. …

基于Stable Diffusion的AIGC服饰穿搭实践

本文主要介绍了基于Stable Diffusion技术的虚拟穿搭试衣的研究探索工作。文章展示了使用LoRA、ControlNet、Inpainting、SAM等工具的方法和处理流程&#xff0c;并陈述了部分目前的实践结果。通过阅读这篇文章&#xff0c;读者可以了解到如何运用Stable Diffusion进行实际操作&…

ScreenToGif-动图制作软件实用操作

ScreenToGif官网&#xff1a;ScreenToGif ⭕第一步&#xff1a;启动页面 ⭕第二步&#xff1a;选项 &#x1f95d;录像机-捕获频率选择手动-播放延迟1000ms(可以任意) ⭕第三步&#xff1a;录像机开始录屏 &#x1f95d;我们调整录屏的大小后&#xff0c;打开画图&#xff0c…

HK1 RBOX X4,Vontar X4,S905 X4 刷 ATV

准备工作 需要HK1 RBOX X4一个&#xff08;内存版本不限 通刷&#xff09;&#xff0c;机顶盒电源&#xff0c;USB双公线一条&#xff08;可以使用两个usb数据线剪开后相同颜色对接使用&#xff0c;最好使用电烙铁焊接一下更稳定&#xff09;&#xff0c;安装 INTEL CPU 运行 w…

Empowering Long-tail Item Recommendation through Cross Decoupling Network (CDN)

Empowering Long-tail Item Recommendation through Cross Decoupling Network (CDN) 来源&#xff1a; KDD’2023Google Research 文章目录 Empowering Long-tail Item Recommendation through Cross Decoupling Network (CDN)长尾问题分析CDNItem Memorization and General…

【算法】leetcode 105 从前序与中序遍历序列构造二叉树

题目 输入某二叉树的前序遍历和中序遍历的结果&#xff0c;请构建该二叉树并返回其根节点。 假设输入的前序遍历和中序遍历的结果中都不含重复的数字。 示例 1: Input: preorder [3,9,20,15,7], inorder [9,3,15,20,7] Output: [3,9,20,null,null,15,7]示例 2: Input: pr…

可控生成之GLIGEN原理

🤗关注公众号 funNLPer 快乐白嫖🤗 论文:GLIGEN: Open-Set Grounded Text-to-Image Generation 代码:gligen/GLIGEN 项目地址:GLIGEN demo地址:gligen demo 文章目录 1. 动机2. 模型结构及原理2.1 输入处理2.1.1 Caption Tokens2.1.2 Grounding Tokens2.2 Gated Sel…

应急三维电子沙盘数字孪生系统

一、简介应急三维电子沙盘数字孪生系统是一种基于虚拟现实技术和数字孪生技术的应急管理工具。它通过将真实世界的地理环境与虚拟世界的模拟环境相结合&#xff0c;实现了对应急场景的模拟、分析和决策支持。该系统主要由三维电子沙盘和数字孪生模型两部分组成。三维电子沙盘是…

ROS-5.自定义topic消息格式

自定义topic消息格式 1. 定义消息1.1. 定义msg文件1.2. 在package.xml中添加功能包依赖1.3. 在CMakeList.txt添加编译选项1.4. 编译 2.定义发布者和订阅者2.1 定义发布者2.2. 定义订阅者2.3. 修改CMakeList.txt2.4 编译 3. 使用消息3.1 启动ros主程序3.2. 启动发布者3.3 启动订…

处理时延降低24倍,联通云粒数据引擎优化实践

*作者&#xff1a;郑扬勇&#xff0c;云粒星河数据中台产品负责人 云粒智慧科技有限公司成立于 2018 年 6 月&#xff0c;是中国联通集团混改以来成立的首家合资公司&#xff0c;是中国智慧城市数智化建设者。一直以来&#xff0c;云粒智慧以数字化、智能化、集约化产品为核心&…

[杂谈]-2023年实现M2M的技术有哪些?

2023年实现M2M的技术有哪些&#xff1f; 文章目录 2023年实现M2M的技术有哪些&#xff1f;1、寻找连接2、M2M与IoT3、流行的 M2M 协议 在当今的数字世界中&#xff0c;机器对机器 (M2M) 正在迅速成为标准。 M2M 包括使联网设备能够交换数据或信息的任何技术。 它可以是有线或无…