- 1、原创力文档(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。。
- 2、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载。
- 3、如文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“版权申诉”(推荐),也可以打举报电话:400-050-0827(电话支持时间:9:00-18:30)。
- 4、该文档为VIP文档,如果想要下载,成为VIP会员后,下载免费。
- 5、成为VIP后,下载本文档将扣除1次下载权益。下载后,不支持退款、换文档。如有疑问请联系我们。
- 6、成为VIP后,您将拥有八大权益,权益包括:VIP文档下载权益、阅读免打扰、文档格式转换、高级专利检索、专属身份标志、高级客服、多端互通、版权登记。
- 7、VIP文档为合作方或网友上传,每下载1次, 网站将根据用户上传文档的质量评分、类型等,对文档贡献者给予高额补贴、流量扶持。如果你也想贡献VIP文档。上传文档
查看更多
C11新特性:一般化常量表达式和constexpr
C++11 新特性:一般化常量表达式和 constexpr
作者:?DevBean?日期: 2012 年 05 月 14 日发表评论?(0)查看评论
参考文章:/pcarlini/entry/c_1x_tidbits_introducing_generalized
先看一个简单的例子。
C++ 中有一个常数表示式(constant expression)的概念。比如,3 + 4 这个表达式会在编译期自动生成 7,而且不会有任何副作用。常数表示式是编译器优化的最佳位置。编译器通常在编译期执行优化,并且将生成的值存入生成的程序代码中。另外,在许多其他情形下,C++ 规范都要求使用常数表示式。例如,数组大小的定义以及枚举值。然而在 C++ 11 之前的版本中,常数表示式在函数调用时却有了额外的问题。比如:
int getFive()
{
return 5;
}
?
int some_value[getFive() + 5]; // Error!
上面的代码中,最后一句是非法的。尽管函数 getFive() 的返回值就是一个常量,但编译器并不能检测出来。它会认为该函数返回值是一个变量,因此报错。
再来看另外的一段代码:
templateint M class F { };
?
Fstd::numeric_limitsint::max() f; // Error!
当然,上面的代码也是最后一行出错。numeric_limits::max?是 C++ 03 标准库函数,其返回值不能用于模板类 F。然而,C 风格的宏也不能用于 C++,例如?INT_MAX。如果我们要实现类似意思的代码,就只能按照如下方式进行:
const int z = numeric_limitsint::max();
此时,我们可以将?z?应用于上述定义中。但是,这样定义的 z 是动态的(在运行时),而不能静态初始化。但是,按照我们的直觉,max()?函数的返回值当然可以作为一个“常量”,它是能够在编译期确定的。一般化的常量表达式,也就是这种由“足够简单的”函数生成的常量表达式,应该有更清晰的语义。这种想法也有利于改进类型安全机制(例如,去除一些常量宏);牺牲编译时间获取代码的可移植性;改进系统编程和泛型编程的支持。
在当前版本 GCC 的运行时库的某些函数(并不是 C++ 11 的新特性),例如 max(),有一个新的关键字constexpr。例如:
static constexpr int max()
{
?return __INT_MAX__;
}
这样,max() 就声明为一个 constexpr 函数。只有足够简单的函数(例如,仅有一行 return 语句,没有循环,不改变参数值等)可以被声明为 constexpr 的。其后果是,参数必须为常量表达式(因此才能在编译期运行函数),而函数本身则在运行时就被执行,其返回值直接进入到生成的代码。
下面是另外一个例子:
constexpr int square(int x)
{
return x * x;
}
?
constexpr int abs(int x)
{
return x 0 ? -x : x;
}
?
constexpr int fac(int x)
{
return x 2 ? x * fac(x - 1) : 1;
}
?
float array[square(9)]; // Ok (not C99 VLA)
?
std::bitsetabs(-87) s; // Ok
?
enum { Max = fac(5) }; // Ok
注意,在最新的标准中,递归是循环的。另外,下面的代码
extern const int medium;
const int high = square(medium); // Ok, dynamic init
是合法的,但是调用?square()?函数却等价于一个普通的函数调用。因此,high 实际还是在运行时初始化的,因为在编译期,medium 的值是不知道的。这并不是 C++ 03 的常量表达式,而是另外的东西。
在 C++ 11 中,也有这么一个常量表达式的概念:
constexpr int s = square(5); // Ok
constexpr int high = square(medium); // error!
也有常量表达式构造函数的概念:
struct complex
{
constexpr complex(double r, double I): re(r), im(i) {}
constexpr double real() { return re; }
constexpr
您可能关注的文档
最近下载
- 急救知识培训:心肺复苏操作流程培训.pptx VIP
- 2024春部编版五年级下册语文教案.docx VIP
- 人教人音版五年级音乐上册《大家一起来》优品课件.ppt VIP
- 第2课 中国人首次进入自己的空间站(1课时)-课件(共23张PPT).pptx VIP
- Microsoft Word - 荧光实验 分子荧光光谱法测定微量铁的含量.pdf VIP
- 2022版三甲评审资料 医院安全保卫管理制度.pdf VIP
- 幼儿园小学生零食与健康ppt 家长进课堂助教主题班会课件.pptx VIP
- 医院中层干部职务任免工作管理办法.PDF
- “三重一大”事项决策实施细则.docx VIP
- 保护区安全准入复习试题附答案.doc
文档评论(0)