- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
背景
这是受到这个问题/答案以及随后在评论中的讨论的启发:Is the definition of “volatile” this volatile, or is GCC having some standard compliancy problems? .根据其他人和我对应该发生的事情的解释,如评论中所述,我已将其提交给 GCC Bugzilla:https://gcc.gnu.org/bugzilla/show_bug.cgi?id=71793仍然欢迎其他相关回复。
此外,该线程已经引起了这个问题:Does accessing a declared non-volatile object through a volatile reference/pointer confer volatile rules upon said accesses?
介绍
我知道 volatile
不是大多数人认为的那样,而是一个实现定义的毒蛇巢。而且我当然不想在任何实际代码中使用以下结构。也就是说,我对这些例子中发生的事情完全感到困惑,所以我真的很感激任何解释。
我的猜测是这是由于对标准的高度细微的解释或(更有可能?)只是所用优化器的极端情况。无论哪种方式,虽然更学术而不是实际,但我希望这被认为是有值(value)的分析,特别是考虑到通常被误解的情况 volatile
是。更多的数据点——或者更有可能的是,反对它的点——一定是好的。
输入
鉴于此代码:
#include <cstddef>
void f(void *const p, std::size_t n)
{
unsigned char *y = static_cast<unsigned char *>(p);
volatile unsigned char const x = 42;
// N.B. Yeah, const is weird, but it doesn't change anything
while (n--) {
*y++ = x;
}
}
void g(void *const p, std::size_t n, volatile unsigned char const x)
{
unsigned char *y = static_cast<unsigned char *>(p);
while (n--) {
*y++ = x;
}
}
void h(void *const p, std::size_t n, volatile unsigned char const &x)
{
unsigned char *y = static_cast<unsigned char *>(p);
while (n--) {
*y++ = x;
}
}
int main(int, char **)
{
int y[1000];
f(&y, sizeof y);
volatile unsigned char const x{99};
g(&y, sizeof y, x);
h(&y, sizeof y, x);
}
g++
来自
gcc (Debian 4.9.2-10) 4.9.2
(Debian
stable
又名 Jessie) 与命令行
g++ -std=c++14 -O3 -S test.cpp
为
main()
生成以下 ASM .版本
Debian 5.4.0-6
(当前
unstable
)产生等效的代码,但我碰巧先运行旧的,所以这里是:
main:
.LFB3:
.cfi_startproc
# f()
movb $42, -1(%rsp)
movl $4000, %eax
.p2align 4,,10
.p2align 3
.L21:
subq $1, %rax
movzbl -1(%rsp), %edx
jne .L21
# x = 99
movb $99, -2(%rsp)
movzbl -2(%rsp), %eax
# g()
movl $4000, %eax
.p2align 4,,10
.p2align 3
.L22:
subq $1, %rax
jne .L22
# h()
movl $4000, %eax
.p2align 4,,10
.p2align 3
.L23:
subq $1, %rax
movzbl -2(%rsp), %edx
jne .L23
# return 0;
xorl %eax, %eax
ret
.cfi_endproc
volatile
局部变量在堆栈上这样做的原因相当明显。但那是他们唯一分享的东西......
f()
确保从 x
读取在每次迭代中,大概是由于它的 volatile
- 但只是将结果转储到 edx
,大概是因为目的地y
未声明 volatile
并且永远不会被读取,这意味着在 as-if 规则下可以禁止对其进行更改。好的,有道理。volatile
真正用于硬件寄存器,显然本地值不能是其中之一 - 否则不能在 volatile
中修改除非它的地址被传递出去......事实并非如此。听着,volatile
没有太多意义。本地值(value)观。但是 C++ 允许我们声明它们并尝试用它们做一些事情。就这样,我们一如既往地迷茫,蹒跚前行。 g()
: 什么。通过移动 volatile
source 转换为传值参数,它仍然只是另一个局部变量,GCC 以某种方式决定它不是或更少 volatile
,所以它不需要每次迭代都读取它......但它仍然执行循环,尽管它的主体现在什么都不做。 h()
: 取已通过 volatile
作为传递引用,与 f()
的有效行为相同已恢复,因此循环执行 volatile
读。f()
的原因,仅此案例对我来说就具有实际意义。 .详细说明:想象一下x
指的是硬件寄存器,每次读取都会产生副作用。你不会想跳过其中任何一个。 #define volatile /**/
导致
main()
正如你所期望的那样,是一个空运。所以,当存在时,即使是局部变量
volatile
确实做了一些事情......我只是不知道在
g()
的情况下是什么情况| .那里到底发生了什么?
volatile
.也没有地址传递出去 - 也没有 static
地址,排除任何内联 ASM POKE
ry - 所以它们永远不能在函数之外被修改。编译器可以看到每个都是常量,永远不需要重新读取,并且volatile
只是不是真的 -volatile
)- volatile
局部变量更多 volatile
相对于其它的? g()
是人们多年来梦寐以求的手动计时循环?)如果我们得出结论,其中任何一项都没有标准影响,我会将其移至他们的 Bugzilla,仅供引用。
最佳答案
对于 f:GCC 消除了非 volatile 存储(但不是加载,如果源位置是内存映射硬件寄存器,则可能会产生副作用)。这里真的没有什么令人惊讶的。
对于 g:因为 x86_64 ABI参数 x
的 g
在寄存器中分配(即 rdx
)并且在内存中没有位置。读取通用寄存器没有任何可观察到的副作用,因此消除了死读。
关于c++ - 为什么 volatile 局部变量的优化与 volatile 参数不同,为什么优化器会从后者生成无操作循环?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38235112/
我是 PHP 新手。我一直在脚本中使用 for 循环、while 循环、foreach 循环。我想知道 哪个性能更好? 选择循环的标准是什么? 当我们在另一个循环中循环时应该使用哪个? 我一直想知道要
我在高中的编程课上,我的作业是制作一个基本的小计和顶级计算器,但我在一家餐馆工作,所以制作一个只能让你在一种食物中读到。因此,我尝试让它能够接收多种食品并将它们添加到一个价格变量中。抱歉,如果某些代码
这是我正在学习的一本教科书。 var ingredients = ["eggs", "milk", "flour", "sugar", "baking soda", "baking powder",
我正在从字符串中提取数字并将其传递给函数。我想给它加 1,然后返回字符串,同时保留前导零。我可以使用 while 循环来完成此操作,但不能使用 for 循环。 for 循环只是跳过零。 var add
编辑:我已经在程序的输出中进行了编辑。 该程序要求估计给定值 mu。用户给出一个值 mu,同时还提供了四个不等于 1 的不同数字(称为 w、x、y、z)。然后,程序尝试使用 de Jaeger 公式找
我正在编写一个算法,该算法对一个整数数组从末尾到开头执行一个大循环,其中包含一个 if 条件。第一次条件为假时,循环可以终止。 因此,对于 for 循环,如果条件为假,它会继续迭代并进行简单的变量更改
现在我已经习惯了在内存非常有限的情况下进行编程,但我没有答案的一个问题是:哪个内存效率更高;- for(;;) 或 while() ?还是它们可以平等互换?如果有的话,还要对效率问题发表评论! 最佳答
这个问题已经有答案了: How do I compare strings in Java? (23 个回答) 已关闭 8 年前。 我正在尝试创建一个小程序,我可以在其中读取该程序的单词。如果单词有 6
这个问题在这里已经有了答案: python : list index out of range error while iteratively popping elements (12 个答案) 关
我正在尝试向用户请求 4 到 10 之间的整数。如果他们回答超出该范围,它将进入循环。当用户第一次正确输入数字时,它不会中断并继续执行 else 语句。如果用户在 else 语句中正确输入数字,它将正
我尝试创建一个带有嵌套 foreach 循环的列表。第一个循环是循环一些数字,第二个循环是循环日期。我想给一个日期写一个数字。所以还有另一个功能来检查它。但结果是数字多次写入日期。 Out 是这样的:
我想要做的事情是使用循环创建一个数组,然后在另一个类中调用该数组,这不会做,也可能永远不会做。解决这个问题最好的方法是什么?我已经寻找了所有解决方案,但它们无法编译。感谢您的帮助。 import ja
我尝试创建一个带有嵌套 foreach 循环的列表。第一个循环是循环一些数字,第二个循环是循环日期。我想给一个日期写一个数字。所以还有另一个功能来检查它。但结果是数字多次写入日期。 Out 是这样的:
我正在模拟一家快餐店三个多小时。这三个小时分为 18 个间隔,每个间隔 600 秒。每个间隔都会输出有关这 600 秒内发生的情况的统计信息。 我原来的结构是这样的: int i; for (i=0;
这个问题已经有答案了: IE8 for...in enumerator (3 个回答) How do I check if an object has a specific property in J
哪个对性能更好?这可能与其他编程语言不一致,所以如果它们不同,或者如果你能用你对特定语言的知识回答我的问题,请解释。 我将使用 c++ 作为示例,但我想知道它在 java、c 或任何其他主流语言中的工
这个问题不太可能帮助任何 future 的访问者;它只与一个小的地理区域、一个特定的时间点或一个非常狭窄的情况有关,这些情况并不普遍适用于互联网的全局受众。为了帮助使这个问题更广泛地适用,visit
我是 C 编程和编写代码的新手,以确定 M 测试用例的质因数分解。如果我一次只扫描一次,该功能本身就可以工作,但是当我尝试执行 M 次时却惨遭失败。 我不知道为什么 scanf() 循环有问题。 in
这个问题已经有答案了: JavaScript by reference vs. by value [duplicate] (4 个回答) 已关闭 3 年前。 我在使用 TSlint 时遇到问题,并且理
我尝试在下面的代码中添加 foreach 或 for 循环,以便为 Charts.js 创建多个数据集。这将允许我在此折线图上创建多条线。 我有一个 PHP 对象,我可以对其进行编码以稍后填充变量,但
我是一名优秀的程序员,十分优秀!