C++中论在类中成员变量定义顺序的重要性

1.问题引入以及段错误介绍

今天,我在运行代码的时候,突然报出了一个错误,那就是Segmentation fault (core dumped) ,也就是段错误(核心转储)。

相信大家大多碰见过这个错误。

段错误(SIGSEGV)是程序试图访问非法内存地址时,操作系统强制终止程序并生成的崩溃报告。它通常意味着你的代码存在以下问题:

  1. 空指针 / 野指针访问 :解引用了一个 nullptr,或者访问了已经释放的内存。
  2. 数组越界:访问了超出数组 / 缓冲区范围的内存。
  3. 栈溢出:函数调用过深,或在栈上分配了过大的局部变量。
  4. 多线程内存冲突:多个线程同时修改同一块未加锁保护的数据。

2.类的成员变量初始化而导致出现段错误

使用gdb调试后,我才发现原来是一个数组越界访问,知道运行到该数组在构造函数初始化的时候,我才发现竟然是我的成员变量定义顺序 有错而导致的(该问题我之前见过,但是此次遇到了才写一篇警醒)。

我们来看一下下面这个代码

cpp 复制代码
#include <iostream>
#include <memory>  // 智能指针头文件
using namespace std;

class SmartArray {
private:
    // 错误的定义顺序:数组先定义,大小后定义
    unique_ptr<int[]> arr;  // 1. 先初始化
    int arr_size;           // 2. 后初始化

public:
    // 初始化列表书写顺序不影响实际初始化顺序!
    SmartArray(int size)
        : arr_size(size),          // 书写顺序第一,但实际第二执行
          arr(make_unique<int[]>(arr_size))  // 书写顺序第二,但实际第一执行
    {}

    void fill() {
        for (int i = 0; i < arr_size; ++i) {
            arr[i] = i * 10;  // 段错误!arr 是大小为 0 的数组
        }
    }

    void print() {
        for (int i = 0; i < arr_size; ++i) {
            cout << arr[i] << " ";
        }
        cout << endl;
    }
};

int main() {
    SmartArray obj(5);
    obj.fill();  // 代码会崩溃
    return 0;
}

这个代码就是典型的成员变量顺序定义错误,由于我们在析构的时候,如果我们用SmartArry():这种格式初始化我们的成员变量,我们执行顺序并不是按照这里面的书写顺序,而是按照我们的成员变量定义顺序

3.此问题后,我们的结论。

对于我们要在一个类中定义一个数组,且定义一个数组大小的变量,那么我们一定要把数组大小的成员变量先定义在数组前面

相关推荐
拳里剑气1 小时前
C++算法:前缀和
开发语言·c++·算法·前缀和
啊我不会诶1 小时前
Codeforces Round 1091 (Div. 2) and CodeCraft 26
c++·算法
H Journey1 小时前
常用知识总结C++、CMake、Linux
linux·c++·opencv·cmake
cany10001 小时前
C++ -- 宏和模板
开发语言·c++
初心未改HD2 小时前
Go语言接口与nil深度解析
开发语言·golang
Achou.Wang2 小时前
go语言并发编程
java·开发语言·golang
小王师傅662 小时前
【Java结构化梳理】泛型-初步了解-中
java·开发语言
CQU_JIAKE2 小时前
[q]4.25
java·开发语言·前端
涵涵(互关)2 小时前
语法大全-only-writer
开发语言·前端·vue.js·typescript