C++> const和constexpr区别与联系

提到const和constexpr,就需要引入常量表达式。常量表达式是指值不会改变,并且编译过程就能得到计算结果的表达式。 => 编译阶段就能得到值,并且不能改变。

const修饰对象无法修改,constexpr更侧重于修饰对象编译期确定且无法修改。具体区别,体现在以下两个方面:

修饰变量

const变量,表示一个变量无法改变,但初值并不确定,不能在编译阶段决定。
比如,

const int sz = get_size(); // 虽然sz无法改变,但get_size() 编译阶段无法确定值,也就是说sz不是常量表达式

constexpr变量,编译器在编译阶段验证变量是否为一个常量表达式。
constexpr侧重变量初值编译阶段确定,且无法修改。如果认定变量是一个常量表达式,就把它声明称constexpr类型。

constexpr int mf = 20; // 字面量20是常量表达式
constexpr int limit = mf + 1; // mf + 1是常量表达式
constexpr int sz = size(); // 只有当size是constexpr函数时,才是正确的

修饰指针

const修饰指针分为两种情况:顶层const, 底层const。
顶层const代表指针变量自身无法修改;底层const代表指针所指对象无法修改。

int i = 10;
int *const p1 = &i; // 顶层const
const int *p2 = &i; // 底层const

p1 = new int(20); // 错误,顶层const指针自身无法修改
p2 = new int(30); // 正确,底层const指针可以修改
*p1 = 40; // 正确,顶层const指针指向的对象可以修改
*p2 = 40; // 错误,底层const指针指向的对象无法修改

constexpr修饰指针,仅对指针有效,与指针所指对象无关

// j的定义必须放在函数体外
int j = 30;

// 函数体内
constexpr int *pp1 = &j; // 等价于 int constexpr *pp1 = &j;
cout << *pp1 << endl; // 30
*pp1 = 40;
cout << j << endl; // 40
pp1 = nullptr; // 错误,constexpr指针无法修改

修饰函数

  • const修饰成员函数,通常称为const函数,表示该函数不会修改类的状态(即不会通过任何方式修改类数据成员)。另外,const类对象,只能调用const函数,确保不会修改类的数据成员
  • constexpr无法修饰成员函数,只能作为函数返回值类型,表明该函数返回的是一个编译期可确定的常量;constexpr 被隐式隐式指定为内联函数,只能在类的声明中定义(.h文件)。

参见C++的const类成员函数

// A.h
class A{
public:
  A():curSize(10) {}
  void setSize(int size) { size++; }

  // const函数
  int size() const { return curSize; }  // 正确示例:不写任何数据成员
  int size() const { curSize = 2; return curSize; }; // 错误示例: const函数不能修改任何类的数据成员
  int size() const { setSize(); return curSize; } // 错误示例:const函数不能调用任何可能导致类的数据成员改变的函数,也就是说,如果调用自身成员函数,只能调用const函数
  const int size() { return curSize; }  // 函数返回值为const类型:函数体可以修改数据成员,但返回类型是const,也就是调用者无法修改

  // 函数返回值为constexpr类型
  constexpr int getMaxSize() { return INT_MAX; } // 正确示例:返回常量值
  // 错误示例:vec.size()运行时确定,不能在编译期决定
  constexpr int getMaxSize() 
  {
    std::vector<int> vec;
    vec.push_back(1);
    vec.push_back(2);
    return vec.size();
  }
  // 正确:虽然看起来返回的是变量,但编译器可确定
  constexpr int getMaxSize(int a, int b) 
  {
    return a + b;
  }

private:
  int curSize;
}

// A.cpp
// 错误示例:constexpr被隐式指定为内联函数,只能在.h 类内定义,不能在类实现文件(.cpp)中定义
constexpr int A::getMaxsize() 
{
  return 1;
}
posted @ 2021-03-17 16:28  明明1109  阅读(1461)  评论(0编辑  收藏  举报