在 C 语言中,我有这段代码:
int a;
a = 10 + 5 - 3
我想问一下:(10+5-3)存储在哪里?
(据我所知,a
存储在栈中,那么 (10+5-3)
呢?这个右值是如何计算的?)
我想问一下:(10+5-3)存储在哪里?
(据我所知,a
存储在栈中,那么 (10+5-3)
呢?这个右值是如何计算的?)
在 C 语言中,我有这段代码:
int a;
a = 10 + 5 - 3
我想问一下:(10+5-3)存储在哪里?
(据我所知,a
存储在栈中,那么 (10+5-3)
呢?这个右值是如何计算的?)
我想问一下:(10+5-3)存储在哪里?
(据我所知,a
存储在栈中,那么 (10+5-3)
呢?这个右值是如何计算的?)
mov <some dest, typically DS-relative>, $0C
$0C是“r-value”。
如果r-value恰好是在运行时进行的计算结果,例如底层c代码是:a = 17 * x; //x是一些运行时变量,则r-value也会作为程序二进制文件中的一系列指令“存储”(或者更确切地说是实现)。与上面简单的“mov dest,imm”的区别在于,加载变量x到累加器中,乘以17,并将结果存储在变量a所在的地址需要多个指令。编译器可能会授权自己使用堆栈来保存某些中间结果等,但这完全取决于编译器,而且通常只涉及r-value的一部分。因此,可以说r-value是一个编译时的概念,封装在程序的某些部分中(而不是数据),并且没有存储在任何地方,除了程序二进制文件。
针对paxdiablo的回应:上述解释确实限制了可能性,因为C标准实际上并没有规定任何这样的东西。即使如此,几乎所有的r-value最终都会通过设置一些指令部分地实现,以便正确地寻址适当的值,无论是计算得出的(在运行时)还是立即获得的。
常量在编译时可能会被简化,因此您所提出的问题可能无法帮助。但是,像 i - j + k
这样需要从一些变量在运行时计算的表达式,可能会被编译器存储在 CPU 架构允许的任何位置:编译器通常会尽力使用寄存器。
LOAD AX, i
SUB AX, j
ADD AX, k
在计算这样的表达式时,可以将其“存储”在累加器寄存器AX中,然后再使用STORE AX, dest
或类似语句将其赋值给某个内存位置。如果现代优化编译器在一个相当简单的表达式上需要将寄存器溢出到内存中,即使是在一个半好的CPU架构(是的,包括x86!-),我会非常惊讶!
a
的汇编指令的操作数。这里是 MSVC 的反汇编代码:
int a;
a = 10 + 5 - 3;
0041338E mov dword ptr [a],0Ch
它存储的位置实际上完全取决于编译器。标准没有规定这种行为。
一个典型的位置可以通过实际编译代码并查看汇编输出来看到:
int main (int argc, char *argv[]) {
int a;
a = 10 + 5 - 3;
return 0;
}
生成结果如下:
.file "qq.c"
.def ___main;
.scl 2;
.type 32;
.endef
.text
.globl _main
.def _main;
.scl 2;
.type 32;
.endef
_main:
pushl %ebp
movl %esp, %ebp
subl $8, %esp
andl $-16, %esp
movl $0, %eax
addl $15, %eax
addl $15, %eax
shrl $4, %eax
sall $4, %eax
movl %eax, -8(%ebp)
movl -8(%ebp), %eax
call __alloca
call ___main
movl $12, -4(%ebp) ;*****
movl $0, %eax
leave
ret
相关部分标记为;*****
,您可以看到该值是由编译器创建并直接插入到mov
类型指令中的。
请注意,之所以这么简单,是因为表达式是一个常量值。一旦引入非常量值(如变量),代码就会变得更加复杂。这是因为您必须在内存中查找这些变量(或它们可能已经在寄存器中),然后在运行时而不是编译时操作这些值。
至于编译器如何计算值应该是什么,那就涉及到表达式求值,这是另一个问题:-)
这取决于编译器。通常值(12)将由编译器计算。然后将其存储在代码中,通常作为加载/移动立即汇编指令的一部分。
你的问题基于错误的前提。
C 语言中,左值(lvalue)的定义特性是它具有存储位置,即被储存。这是左值(lvalue)与右值(rvalue)的区别所在。右值(rvalue)并没有被储存的地方。这就是它成为右值(rvalue)的原因。如果它被储存了,那么根据定义,它就是左值(lvalue)。
0
初始化变量时,机器代码可能只是将一个寄存器与自身进行xor
操作。在这种情况下,“0”是否“存储在某处”?不需要。可以通过先前清零的寄存器的增量生成“1”。等等。R值不必存储在任何地方。这就是使它们成为r值的原因。 - AnT stands with Russia(10+5-3)
是一个表达式,恰好是一个rvalue(因为您无法对其应用&运算符--在C++中规则更加复杂)。在运行时,没有表达式、lvalue或rvalue。特别地,它们不会被存储在任何地方。12
相反,后者将是一个rvalue,但12
不出现在您的程序中)。