内容简介:Clang and GCC (and others) support function attributesTheConsider:
Clang and GCC (and others) support function attributes __attribute__((const))
and __attribute__((pure))
. These had always been explained to me thus:
-
__attribute__((pure)): The function may read, but doesn’t modify any global state. [This is true!] -
__attribute__((const)): The function neither reads nor modifies global state. [ This is MASSIVELY OVERSIMPLIFIED to the point of being a lie!]
The
gigantic
asterisk on that simplified explanation of __attribute__((const))
is that pointers and references—even const pointers and references!—are disallowed if the memory they reference can change between successive invocations of the function.
Consider:
bool ends_with(const string & full, const string & suffix) __attribute__((const));
The implementation of this should certainly meet the requirements of neither reading nor writing global state… unless you consider the memory referenced by your parameters to be global state!
But that’s exactly what GCC’s docs on the attribute do consider as global state:
Note that a function that has pointer arguments and examines the data pointed to must not
be declared const
if the pointed-to data might change between successive invocations of the function. In general, since a function cannot distinguish data that might change from data that cannot, const functions should never take pointer or, in C++, reference arguments.
(Note that Clang, despite having supported the attribute for years, has zero docs on it. Based on experience, though, it seems to have to the same restriction.)
The consequence of violating this requirement is that in the following code, result2
might
wind up being true (if you get unlucky with the optimizer):
string s = "foo"; bool result1 = ends_with(s, "oo"s); // True, always s = "bar"; bool result2 = ends_with(s, "oo"s); // True, maybe!
That’s a somewhat contrived example, but it can be much more insidious when it’s a member variable you’re passing to your __attribute__((const))
function. This can also bite you when you have implicit
conversions to const ref (e.g., when you’re passing a string literal to a function that converts it to a const string reference).
The worst part of all this—and the reason I’ve spent 2 full days over the course of the last few months debugging this—is that neither the compiler nor UBsan provide any warnings that this is happening.
If you squint really hard, you can kind of see what the compiler implementers were thinking—what you’re “really” passing here is just a pointer, which (again if you squint hard enough) is really just an integer… it’s your fault, silly programmer, for turning looking at the (“global”) memory state corresponding to that int.
This is what people mean when they say C++ is a language of footguns.
以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,也希望大家多多支持 码农网
猜你喜欢:本站部分资源来源于网络,本站转载出于传递更多信息之目的,版权归原作者或者来源机构所有,如转载稿涉及版权问题,请联系我们。
机器学习算法原理与编程实践
郑捷 / 电子工业出版社 / 2015-11 / 88.00
本书是机器学习原理和算法编码实现的基础性读物,内容分为两大主线:单个算法的原理讲解和机器学习理论的发展变迁。算法除包含传统的分类、聚类、预测等常用算法之外,还新增了深度学习、贝叶斯网、隐马尔科夫模型等内容。对于每个算法,均包括提出问题、解决策略、数学推导、编码实现、结果评估几部分。数学推导力图做到由浅入深,深入浅出。结构上数学原理与程序代码一一对照,有助于降低学习门槛,加深公式的理解,起到推广和扩......一起来看看 《机器学习算法原理与编程实践》 这本书的介绍吧!