快速初始化成员变量

本文介绍了C++中成员变量初始化的不同方式,包括C++98和C++11的区别。在C++98中,只能初始化静态整形常量或枚举类型;而在C++11中,可以使用=或{}

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

在C++98中,可以在头文件类的声明中使用“=”来给成员变量初始化,但是要求却很苛刻,被初始化的成员变量必须是:静态整形常量或者静态枚举类型。

 1 class Init {
 2 public:
 3     Init():a(0){}
 4     Init(int d):a(d){}
 5 
 6 private:
 7     int a;
 8     const static int b = 0;
 9     int c = 1;    // 成员,无法通过编译
10     static int d = 0;    // 成员,无法通过编译
11     static const double e = 1.3;    // 非整型或枚举类型,无法通过编译
12     static const char *const f = "e";    // 非整型或枚举类型,无法通过编译
13 };

g++编译器对于第11行和12行的静态常量就地初始化是支持的,但是这只是GNU对C++的扩展,并不是标准C++。

在C++ 11中,允许非静态成员变量的初始化有多种形式。除了初始化列表外,在C++ 11中,还允许使用等号=或者花括号{}进行就地初始化。比如:

1 struct Init
2 {
3     int a = 1;
4     double d{1.2};
5     string s("Hello"); // 无法通过编译,{}与()是不等价的
6 };

C++ 11标准支持了就地初始化非静态成员变量,同时,初始化列表也被保留了下来。如果两者都使用,则初始化列表的优先级更高。

 1 #include <iostream>
 2 using namespace std;
 3 
 4 struct Mem {
 5     Mem() {cout << "Mem default, num: " << num << endl;}
 6     Mem(int i):num(i) {cout << "Mem, num: " << num << endl;}
 7     int num = 2;    // 使用=初始化非静态成员变量
 8 };
 9 
10 class Group {
11 public:
12     Group() {cout << "Group default, val:" << val << endl;}
13     Group(int i) val('G'), a(i) {cout << "Group, Var: " << val << endl;}
14     void NumofA() {cout << "number of A: " << a.num << endl;}
15     void NumofB() {cout << "number of B: " << b.num << endl;}
16 
17 private:
18     char val{'g'}; // 使用{}初始化非静态成员
19     Mum a;
20     Mum b{19};    // 使用{}初始化非静态成员
21 };
22 
23 int main()
24 {
25     Mem member;    // Mem default, num: 2
26     Group group; // Mem default, num: 2
27                  // Mem, num: 19
28                  // Group default, val: g
29 
30     group.NumofA();    // number of A: 2
31     group.NumofB();    // number of B: 19
32 
33     Group group2(7); // Mem, num: 7
34                      // Mem, num: 19
35                      // Group, val: G
36 
37     group2.NumofA(); // number of A: 7
38     group2.NumofB(); // number of B: 19
39     return 0;
40 }

 

转载于:https://www.cnblogs.com/lniwn/p/3387543.html

<think>我们正在使用Qt Creator,并且想要快速初始化成员变量。根据引用[4]的内容,我们知道在Qt Creator中,如果构造函数中没有初始化成员属性,那么成员属性会进行默认初始化。但是,对于指针成员和普通成员,默认初始化的行为可能不同。 在C++中,类的成员变量可以通过构造函数的初始化列表进行初始化。在Qt Creator中,我们可以利用一些快捷方式或者代码片段来快速生成初始化代码。 以下是一些方法: 1. **使用构造函数初始化列表**:这是C++中推荐的方式,可以高效地初始化成员变量。 2. **使用Qt Creator的代码片段**:Qt Creator支持自定义代码片段,我们可以创建代码片段模板来快速生成初始化代码。 3. **使用ClangFormat或Beautifier**:虽然引用[2]和[5]主要讨论代码格式化,但我们可以配置快捷键来快速格式化代码,但这不是初始化成员变量的直接方法。 下面详细介绍: ### 方法1:手动编写初始化列表 在类的构造函数后使用初始化列表,例如: ```cpp MyClass::MyClass(QWidget *parent) : QWidget(parent) , m_member1(0) // 初始化普通成员 , m_member2(nullptr) // 初始化指针成员 { // 构造函数体 } ``` ### 方法2:使用Qt Creator的代码片段(Code Snippets) 我们可以自定义一个代码片段,用于快速生成初始化列表。 步骤: 1. 打开Qt Creator,进入`工具`->`选项`->`文本编辑器`->`片段`。 2. 点击`添加`,创建一个新的代码片段。 3. 设置触发词(例如`initmem`),然后在代码片段中填写: ``` : ${base_class}(${parent}) , ${member1}(${value1}) , ${member2}(${value2}) { ${cursor} } ``` 4. 保存后,在构造函数中输入触发词`initmem`,然后按`Tab`键,即可生成模板,然后填写具体的成员变量名和初始值。 ### 方法3:使用重构功能 Qt Creator提供了重构功能,可以自动生成成员变量初始化。 步骤: 1. 在类中声明成员变量(例如`int m_count;`)。 2. 在构造函数的实现位置(cpp文件中),将光标放在构造函数的参数列表后面。 3. 右键选择`Refactor` -> `Initialize member variable`(或者使用快捷键,取决于你的配置),然后选择要初始化成员变量。 4. 输入初始值,Qt Creator会自动在构造函数的初始化列表中添加该成员的初始化。 注意:这个功能可能依赖于Qt Creator的版本,较新的版本支持得更好。 ### 方法4:使用代码模板 在创建类的时候,使用自定义的类模板,在模板中已经包含了初始化列表的框架。这需要配置Qt Creator的类模板。 步骤: 1. 进入`工具`->`选项`->`C++`->`文件命名`。 2. 在“类模板”中,可以设置自定义模板。例如: ```cpp #ifndef %GUARD% #define %GUARD% class %CURSOR% { public: %CURSOR%(); // 其他成员函数 private: // 成员变量 }; #endif // %GUARD% ``` 但是,这并不能直接初始化成员变量,我们可以在构造函数实现模板中设置初始化列表。 3. 在“文件命名”设置中,选择“源文件模板”,在构造函数实现中写入: ```cpp %CLASS%::%CLASS%() : // 初始化列表 { } ``` 这样每次创建新类时,构造函数实现中就会包含初始化列表的位置。 ### 补充:关于默认初始化 根据引用[4]: - 普通成员变量(如`int`,`double`)如果没有在初始化列表中初始化,则执行默认初始化(内置类型将是不确定值,如果类有默认构造函数则调用默认构造函数)。 - 指针成员变量默认初始化为随机值(即野指针),所以最好在初始化列表中初始化为`nullptr`。 因此,建议总是使用初始化列表初始化所有成员变量。 ### 总结 在Qt Creator中快速初始化成员变量,最有效的方法是: 1. 使用初始化列表手动初始化。 2. 利用代码片段快速生成初始化列表的框架。 3. 使用重构功能自动添加成员变量初始化。 另外,养成良好的编程习惯:在声明成员变量时,考虑在构造函数中初始化它们,特别是指针和内置类型。 如果上述方法不能满足需求,可能需要检查Qt Creator的版本,并查看是否有相关插件支持。 参考引用[4]中关于成员变量默认初始化的说明[^4]。 §§相关问题§§ 1. Qt Creator中如何自定义代码片段? 2. 如何配置Qt Creator的代码格式化工具(如ClangFormat)? 3. C++初始化列表和构造函数体内赋值有什么区别?</think>在 Qt Creator 中,您可以通过以下快捷方式和技巧快速初始化成员变量: ### 方法 1:使用构造函数初始化列表(推荐) 1. **在头文件声明成员变量**: ```cpp // widget.h class Widget : public QWidget { Q_OBJECT public: explicit Widget(QWidget *parent = nullptr); private: int m_counter; // 普通成员 QString* m_textPtr; // 指针成员 double m_value = 0.0; // C++11 直接初始化 }; ``` 2. **在源文件使用初始化列表**: ```cpp // widget.cpp Widget::Widget(QWidget *parent) : QWidget(parent) // 基类初始化 , m_counter(0) // 普通成员初始化 , m_textPtr(nullptr) // 指针初始化为 nullptr // m_value 已在声明处初始化 { // 构造函数体 } ``` **快捷键**:输入 `ctor` + `Tab` 快速生成构造函数框架 ### 方法 2:使用成员初始化快捷模板 1. **创建代码片段**: - `工具` → `选项` → `文本编辑器` → `片段` - 添加新片段: ```plaintext Trigger: initmem Snippet: , ${var}(${value})${cursor} ``` 2. **使用方式**: ```cpp Widget::Widget() : QWidget(parent) initmem // 输入后按 Tab ``` 自动扩展为: ```cpp , var(value) ``` ### 方法 3:批量初始化C++11 特性) ```cpp // C++11 类内直接初始化 class DataModel { public: int id = -1; QString name = "Untitled"; QTimer* timer = nullptr; // 指针初始化为空 QVector<double> values{1.0, 2.0, 3.0}; // 列表初始化 }; ``` ### 方法 4:利用重构功能 1. 右键点击成员变量 → `Refactor` → `Initialize member variable` 2. 选择初始值类型: - 基本类型:输入数值如 `0` - 指针类型:自动建议 `nullptr` - 对象类型:生成默认构造调用 ### 特殊成员处理技巧 1. **静态成员**(引用[3]): ```cpp // 类外初始化(.cpp 文件) double ClassName::staticVar = 0.0; ``` 2. **指针成员安全初始化**(引用[4]): ```cpp class SafeClass { public: SafeClass() : resourcePtr(new Resource()) // 直接初始化 {} ~SafeClass() { delete resourcePtr; } private: Resource* resourcePtr = nullptr; // 双重保障 }; ``` 3. **Qt 容器初始化**: ```cpp Widget::Widget() : m_list({"Item1", "Item2"}) // C++11 统一初始化 , m_map({{1, "A"}, {2, "B"}}) {} ``` ### 配置建议 1. **代码格式化**(引用[5]): - 设置 `Ctrl+Alt+K` 格式化当前文件 - 确保初始化列表格式一致 2. **启用 C++11**: 在 `.pro` 文件中添加: ```qmake CONFIG += c++11 ``` > **重要提示**:根据引用[4],未显式初始化成员变量: > - 基本类型(int/double)将包含随机值 > - 指针类型将成为野指针 > - 类类型调用默认构造函数 > **务必显式初始化所有成员**[^4]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值