复习 C++ const (二) constexpr和常量表达式,constexpr函数

constexpr和常量表达式

2.5 constexpr和常量表达式

常量表达式:
不会改变且在编译过程中就能得到计算结果

常量表达式:
字面值,用常量表达式初始化的const对象也是常量表达式

const int max_files = 20;
const int max_fils_2 = max_files+1;
int s = 1;//不是常量表达式
const int sz = get_size();//sz不是常量表达式,因为具体值到运行时才能得到,所以不符合在编译中就得到结果这个条件

C++11规定可以用constexpr来验证变量的值是否是一个常量表达式,必须用常量表达式初始化

C++11还允许定义constexpr函数,用于初始化constexpr变量

使用到的类型需要是字面值类型

:算术类型,引用,指针是字面值类型

自定义类,IO库,string等不是字面值类型

constexpr指针初始值必须是nullptr或0,或是存储于某个固定地址中的对象

定义在所有函数体外的对象地址固定不变

阅读C++primer 6.1.1节

constexpr声明中定义一个指针,仅对指针有效

const int *p = nullptr;//指向整型常量的指针
constexpr int * q = nullptr;//常量指针

constexpr会将定义的对象置为顶层const

constexpr指针可以指向常量也可以指向非常量

constexpr int *np = nullptr;
int j = 0;
constexpr int i = 42;//i,j都应定义在所有函数外
constexpr const int*p = &i;
constexpr int *p1 = &j;

C++ primer 第59页有说到:函数体内定义的变量一般不存放在固定地址中,而constexpr指针的初始值必须为nullptr或0,或是存储在某个固定地址中的对象,所以constexpr指针不能指向定义的函数体内的变量(但是static关键字修饰的变量由于函数体结束仍然存在(程序结束才销毁),有固定地址,所以是个例外)

constexpr函数

约定:函数只能有一条return语句 返回类型和所有形参类型都应该是字面值类型

constexpr函数会被隐式的指定为内联函数,在编译过程中随时展开

函数中可以有空语句,类型别名,using声明(运行时不执行任何操作就可以)

constexpr size_t scale(size_t cnt){ return new_sz()*cnt;}
//对于此函数,如果实参为常量表达式,则返回值也是常量表达式,反之不是
(允许constexpr函数的返回值不是常量)
int arr[scale(2)];
int i = 2;
int a2[scale(i)];//错误,scale(i)不是常量表达式,但是运行时没有报错

一般将内联函数和constexpr函数定义在头文件中,这两种函数可以在程序中多次定义

但是多个定义必须完全一致

posted @ 2021-10-02 18:16  ziggystardust  阅读(114)  评论(3)    收藏  举报