常量表达式和constexpr(c++11)
常量表达式
常量表达式是指值不会改变且在编译阶段就能得到计算结果的表达式(两点要求)
const int max_files = ; //是常量表达式
const int limit = max_file + ; //是常量表达式
string s="";
const int siz=s.size(); //不是常量表达式
int staff_size = ; //不是常量表达式
const int sz = get_size(); //不是常量表达式,因为尽管sz本身是一个常量,但是它的值直到程序运行时才能获得
constexpr(c++11)
c++11允许将变量声明为constexpr类型让编译器来验证变量是否为一个常量表达式
constexpr int mf = ;
constexpr int limit = mf + ;
constexpr int sz = get_size(); //只有当get_size()是一个constexpr函数时才是正确的声明
const变量 和 constexpr 变量之间的主要区别
1.const 变量的初始化可以延迟到运行时,而 constexpr 变量必须在编译时进行初始化:
const int sz = get_size(); //正确的声明,但是sz不是常量表达式
constexpr int sz = get_size(); //只有当get_size()是一个constexpr函数时才是正确的声明
2.constexpr声明中如果定义了一个指针,限定符constexpr只对指针有效,与指针所指的对象无关:
const int* p = nullptr; //p是一个指向整型常量的指针
constexpr int* q = nullptr; //q是一个指向整型的常量指针
constexpr const int* r = nullptr; //r是一个指向整型常量的常量指针
constexpr函数
用限定符constexpr修饰函数,使之用于常量表达式中,constexpr函数有这么几个特性:
1.函数的返回类型和所有的形参类型都是字面值类型
2.函数体中只有一条语句,而且只能是一条return语句(但其实也是可以包含其他语句的,只要这些语句在程序运行时不执行任何操作就行,如:空语句、类型别名、using声明)
constexpr int new_sz()
{
return ;
}
3.编译器其实也允许constexpr函数的返回值和形参不是字面值(编译器有时可忽略函数的constexpr属性)
constexpr int get_size(int a)
{
return a;
} int main()
{
int size = ;
int y = get_size(size); //正确,编译器此时忽略函数的constexpr属性,因为y是非constexpr的,可以不要求get_size是constexpr的
constexpr int z = get_size(size); //错误!!! int arr1[get_size()]; //正确
int arr2[get_size(size)]; //错误,数组的大小要求是常量表达式 return ;
}
4.constexpr函数自带inline属性
5.inline函数和constexpr函数可以在程序中多次定义,一般定义在头文件中,但是多个定义必须完全一致:
现在假设我们有2个cpp文件和一个.h头文件:main.cpp(main函数在这个文件里), haha.cpp, xixi.h
main.cpp:
void fuck(){}
int main()
{
return ;
}
haha.cpp:
void fuck(){}
如果只编译main.cpp的话,是不会有重复定义的错误的。但是如果同时编译main.cpp和haha.cpp的话,那就会有重复定义错误。IDE一般都是会默认编译所有文件。
好,那能不能只定义在.h头文件里,然后另外两个cpp文件单纯include?
main.cpp:
#include "xixi.h" int main()
{
return ;
}
haha.cpp:
#include "xixi.h"
xixi.h:
void fuck(){}
结果是也不行,因为include其实就是把xixi.h里面的东西拷到cpp里,结果跟上面那种是一样的,也是在link的时候报错。
好,那我又想了,不是有个什么#ifndef的东西吗?给我的xixi.h加上这个行不行,这样不就只有一个fuck函数的定义了吗?
#ifndef XIXI
#define XIXI
void fuck(){}
#endif
结果还是不行,依然报重复定义错误,所谓的防止头文件重复包含是对应一个cpp文件。也就是说,这个#ifndef可以保证在haha.cpp里,#ifndef里面的内容是唯一的,也能保证在main.cpp里是唯一的,但不能保证你在这个程序里(同时用到main.cpp和haha.cpp)是唯一的。
考虑下面这个例子:
main.cpp:
#include "xixi.h" int main()
{
return ;
}
haha.cpp:
#include "xixi.h"
xixi.h:
#ifndef XIXI
#define XIXI
void fuck(); //这是声明
#endif
xixi.cpp:
#include"xixi.h"
void fuck(){}//定义
这样将不会有重复定义的错误。因为include的头文件里面没有函数定义,只有声明,而同一个函数是可以多次声明。
再回到inline函数的展开,展开需要什么,需要函数体,所以inline函数要定义在头文件里。但最开始的几个错误例子说了,定义在头文件的函数定义被多个cpp文件include后会有重复定义的错误,怎么办呢?用inline关键字可以避免这个问题,指明inline,就不会有这个错误:
main.cpp:
#include "xixi.h" int main()
{
return ;
}
haha.cpp:
#include "xixi.h"
xixi.h:
inline void fuck(){}
这样也不会有重复定义的错误。
综上,inline函数有2个作用:
- 函数展开,避免函数调用开销,(用空间换时间)
- 定义在头文件的inline函数被多个cpp文件include不会产生重复定义(在程序可以被多次定义的意思不是说可以在文件里重复定义,其实是给inline函数做一个标记,假如有10个inline函数体,然后最后会删除剩下1份)
常量表达式和constexpr(c++11)的更多相关文章
- C++常量表达式、const、constexpr(C++11新增)的区别
常量表达式是指值不会改变且在编译过程中就能够得到计算结果的表达式,能在编译时求值的表达式. 程序先编译再运行: 在编译阶段, 编译器将在编译过程中把用到该常量的地方都全都替换为 常量的值. 但是常量 ...
- constexpr和常量表达式
常量表达式:值不会改变并且在编译过程就能得到计算结果的表达式. 字面值属于常量表达式,用常量表达式初始化的const对象也是常量表达式. 一个对象(或表达式)是不是常量表达式由它的数据类型和初始值共同 ...
- 第8课 常量表达式(constexpr)
一. const 和constexpr的区别 (一)修饰变量时,const为“运行期常量”,即运行期数据是只读的.而constexpr为“编译期”常量,这是const无法保证的.两者都是对象和函数接口 ...
- const限定符、constexpr和常量表达式------c++ primer
编译器将在编译过程中把用到const变量的地方都替换成对应的值,为了执行这种替换,编译器必须知道变量的初始值.如果程序包含多个文件,则那个用了const对象的文件都必须能访问到它的初始值才行.要做到这 ...
- C++11特性——变量部分(using类型别名、constexpr常量表达式、auto类型推断、nullptr空指针等)
#include <iostream> using namespace std; int main() { using cullptr = const unsigned long long ...
- C++11 constexpr常量表达式
常量表达式函数 要求: 函数体内只有单一的return返回语句 例如: constexpr int data() { const int i=1; //含有除了return以外的语句 return i ...
- constexpr与常量表达式(c++11标准)
关键字 constexpr 是C++11中引入的关键字,是指值不会改变并且在编译过程中就得到计算结果的表达式.(运行中得到结果的不能成为常量表达式,比如变量). 声明为constexpr的变量一定是一 ...
- C++11常量表达式
[C++11之常量表达式] 关键字:constexpr: 中文学名:常量表达式. constexpr用于把运行期计算放置在编译期. 使用constexpr有3个限制: 1.函数中只能有一个return ...
- constexpr和常量表达式的注意事项
1.常量表达式,是指其值不可改变,且在编译阶段就已经得出计算结果的表达式,例如字面值就是常量表达式. 2.判断是否是常量表达式,要关注数据类型是否是const类型,初始值是否是在编译阶段就得到的. 3 ...
随机推荐
- js中json知识点
首先,json是一种数据格式,而不能说是一种对象(object).这一点是非常重要的. 起源是不同的语言中数据对象的形式是不一样的,我们为了在不同的语言中传递数据,发明了一种json格式用于消除这种差 ...
- gradle项目与maven项目互转
maven to gradle 在maven项目根目录下执行命令: gradle init --type pom 当然你得先下载Gradle,配置完环境变量. gradle to maven grad ...
- vue总结2
1. 给router-link添加事件 之前用v-link 现在用 router-link 添加事件要用原生的.native修饰v-on <my-component v-on:click.nat ...
- bug定位
*:first-child { margin-top: 0 !important; } body>*:last-child { margin-bottom: 0 !important; } /* ...
- Oracle_高级功能(8) 事务和锁
Oracle数据库事务1. 事务定义在数据库中事务是工作的逻辑单元,一个事务是由一个或多个完成一组的相关行为的SQL语句组成,通过事务机制确保这一组SQL语句所作的操作要么都成功执行,完成整个工作单元 ...
- 2019年Java未来的发展方向
2018即将结束,迎来2019年,Java作为世界上 最流行的计算机编程语言,在当今信息时代中发挥了重要的作用.Java语言本身具有着自己独特的优势:面向对象.分布式应用并且安全.多线程.跨平台等.这 ...
- POJ3662或洛谷1948 Telephone Lines
二分答案+单源最短路 POJ原题链接 洛谷原题链接 显然可以二分答案,检验\(mid\)可以使用最短路来解决. 将大于\(mid\)的边看成长度为\(1\)的边,说明要使用免费升级服务,否则长度为\( ...
- python 截取某一天的日志,简单操作
#!/usr/bin/python #Filename: Segmentation_log.py import re,sys def openfile(*args): try: f=open(args ...
- UVA-1364.Knights of the Round Table 无向图BCC
题目链接:https://vjudge.net/problem/UVA-1364 题意:有n个人参加会议,互相憎恨的人不能坐在相邻的位置,并且每个会议参加的人数必须是奇数,求有多少个人不能参加任何一个 ...
- gcc产生类型转换告警
问题背景: 看 https://www.cnblogs.com/sinaxyz/p/4525208.html 这个篇blog时候,发现在应用层代码中,函数 int open_netlink() 中,有 ...