C 语言

关注公众号 jb51net

关闭
首页 > 软件编程 > C 语言 > C++之const和static的使用

C++之const和static的使用及说明

作者:EnigmaCoder

C++中const与static的用法:const限制变量/对象只读性,确保数据安全;static控制变量/函数的存储周期与作用域,实现数据共享,两者结合的static const用于定义类级常量,需注意const变量不可强制修改、static局部变量线程安全问题及类static成员必须类外初始化的要点

本文将分别介绍const与static的用法,最后介绍两者的结合应用和常见坑点。

const:“只读”的守护者

const 的核心作用是限制变量/对象的“可修改性”,强制编译器检查“意外修改”,本质是给代码加“安全锁”。

它的用法围绕“修饰谁”展开,不同修饰对象的含义完全不同,也是初学者最易踩坑的点。

修饰普通变量

示例:

const int max_len = 10; // 正确:定义时初始化
max_len = 20; // 错误:编译器直接报错,禁止修改

修饰指针

const修饰指针关键看 const* 的位置:

int a = 5, b = 10;
const int* p = &a;
*p = 6; // 错误:指向的内容(a的值)不能改
p = &b;  // 正确:指针可以指向新地址(b)
int a = 5, b = 10;
int* const p = &a;
*p = 6; // 正确:指向的内容(a的值)可以改
p = &b;  // 错误:指针不能换指向

修饰函数

修饰函数参数:保证函数内部不修改参数(尤其针对引用/指针参数,避免意外篡改外部变量)。

// 传入字符串,仅读取不修改,用 const 保护
void printStr(const string& s) {
    s += "test"; // 错误:禁止修改 const 参数
    cout << s << endl; // 正确:仅读取
}

修饰函数返回值:限制返回值不可被修改(常见于返回指针/引用的场景,避免外部篡改内部数据)。

// 返回数组首地址,禁止外部修改数组内容
const int* getArr() {
    static int arr[3] = {1,2,3};
    return arr;
}
int main() {
    const int* p = getArr();
    *p = 4; // 错误:返回值是 const,禁止修改
}

修饰类成员

const 成员变量:属于对象的“常量”,必须在构造函数初始化列表中初始化(不能在定义时直接赋值)。

class Person {
private:
    const int age; // const 成员变量
public:
    // 正确:在初始化列表中初始化
    Person(int a) : age(a) {}
    // 错误:不能在构造函数体内赋值
    // Person(int a) { age = a; }
};

const 成员函数:保证函数内部不修改任何非 mutable 成员变量,函数声明和定义都要加 const

class Person {
private:
    int age;
public:
    // 声明时加 const
    void showAge() const;      //语法:返回值类型 函数名() const;
};
// 定义时也要加 const(位置不能错)
void Person::showAge() const {
    age = 20; // 错误:const 函数禁止修改成员变量
    cout << age << endl; // 正确:仅读取
}

修饰对象

const 修饰对象,本质是把对象变成 “只读对象”—— 一旦定义,这个对象的成员变量就不能被修改,能有效保证数据安全(比如防止误改关键数据)。

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

// 定义一个学生类
class Student {
public:
    string name;  // 成员变量:姓名
    int score;    // 成员变量:成绩

    // 构造函数(初始化姓名和成绩)
    Student(string n, int s) : name(n), score(s) {}

    // 普通成员函数:修改成绩(可能改成员变量)
    void setScore(int s) {
        score = s;
    }

    // const 成员函数:只打印信息(不修改成员变量)
    void showInfo() const {
        cout << "姓名:" << name << ",成绩:" << score << endl;
    }
};
int main() {
    // 1. 普通对象(可以修改成员,调用任意函数)
    Student s1("张三", 90); 
    s1.score = 85;          // 允许:普通对象能改成员变量
    s1.setScore(88);        // 允许:普通对象能调用非const函数
    s1.showInfo();          // 允许:普通对象也能调用const函数

    // 2. const对象(只读,核心限制:不能改成员,只能调const函数)
    const Student s2("李四", 85); 
    // s2.score = 90;       // 错误!const对象不能直接修改成员变量
    // s2.setScore(92);     // 错误!const对象不能调用非const函数
    s2.showInfo();          // 允许!const对象只能调用const成员函数

    return 0;
}

遵循两个规则:

static:“静态存储”与“作用域控制”

static 的核心是改变变量/函数的“存储周期”或“作用域”,本质是管理“数据的生命周期”和“访问范围”,常见于“共享数据”“全局唯一”场景。

修饰全局变量

示例:

// a.cpp
static int global_val = 5; // 仅 a.cpp 可见

// b.cpp
extern int global_val; // 错误:无法引用 a.cpp 的 static 全局变量

修饰局部变量

示例:

void countCall() {
    static int cnt = 0; // 仅初始化一次(第一次调用时)
    cnt++;
    cout << "调用次数:" << cnt << endl;
}
int main() {
    countCall(); // 输出 1
    countCall(); // 输出 2(cnt 未被销毁)
}

修饰类成员

static 类成员是“所有对象共享的数据/方法”,不依赖具体对象存在。

static 成员变量

示例:统计类的对象创建个数

class Student {
private:
    static int total; // 声明:属于 Student 类
public:
    Student() { total++; } // 创建对象时计数+1
    ~Student() { total--; } // 销毁对象时计数-1
    // 必须用 static 函数访问 static 变量(见下文)
    static int getTotal() { return total; }
};

// 类外初始化:不加 static,需指定类名
int Student::total = 0;

int main() {
    Student s1, s2;
    cout << Student::getTotal(); // 输出 2(两个对象)
}

static成员函数

const 与 static 结合应用

两者结合的核心场景是“类的静态常量”,即 static const(或 const static,两者在类中等价),用于定义“类级别的常量”(所有对象共用,且不可修改),既实现了数据共享又达到了数据不可被改变的目的。

关键用法:类内静态常量的初始化

C++11 及以后:static const 成员变量可在类内直接初始化(仅限字面量类型,如 int、char 等)。

示例:

class Config {
public:
    // C++11+ 支持:类内直接初始化静态常量
    static const int MAX_NUM = 100;
    static const string DEFAULT_NAME; // 非字面量类型,需类外初始化
};

// 非字面量类型的 static const 成员,仍需类外初始化
const string Config::DEFAULT_NAME = "unknown";

注意:C++11 前,static const 成员变量必须在类外初始化,即使是字面量类型。

避坑指南

坑 1:const 变量能“强制修改”?

用指针强制转换可以修改 const 变量,但这是“未定义行为”(编译器不报错,但运行结果不可控,可能触发崩溃),绝对禁止!

const int a = 5;
int* p = (int*)&a; // 强制转换(危险!)
*p = 10; // 运行时可能修改成功,但属于未定义行为

坑 2:static 局部变量的线程安全?

坑 3:类 static 成员忘记类外初始化?

仅在类内声明 static 成员变量,未在类外初始化,会导致链接错误(编译器找不到变量的定义)。记住:static 类成员“声明在类内,定义在类外”。

结尾总结

conststatic 虽基础,但用法灵活,核心记住两点:

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

您可能感兴趣的文章:
阅读全文