64 位计算机如何更改内存中的一个字节?
How does a 64-bit computer change one byte in memory?
如果 C 程序更改字节数组中的一个字节,会执行哪些机器指令?硬件是否需要读取8个字节,更改一个字节并存储它(使用2次内存操作)?
编辑:专门针对 x86-64 架构
在 x86-64 上,硬件将读取一个 缓存行 ,修改缓存中的字节,最终将缓存行写回内存。
发生回写的主要原因是CPU需要其他数据的缓存行。有强制回写的明确说明,但 C 编译器不太可能使用这些说明。它会减慢 CPU 以强制进行不必要的写入。
这完全取决于编译器、优化等。只需尝试编译和反汇编即可。作为示例,我们将编译以下代码:
#include <stdio.h>
int main() {
char a[] = "01234567890";
a[5] = 'A';
printf("%s\n", a);
}
// gcc -o main -std=c11 -Wall -Wextra -O0 main.c
我们通过objdump
得到反汇编:
6c1: 48 b8 30 31 32 33 34 movabs rax,0x3736353433323130
6c8: 35 36 37
6cb: 48 89 45 ec mov QWORD PTR [rbp-0x14],rax
6cf: c7 45 f4 38 39 30 00 mov DWORD PTR [rbp-0xc],0x303938
6d6: c6 45 f1 41 mov BYTE PTR [rbp-0xf],0x41
6da: 48 8d 45 ec lea rax,[rbp-0x14]
6de: 48 89 c7 mov rdi,rax
// objdump -d ./main -Mintel | less
如果 C 程序更改字节数组中的一个字节,会执行哪些机器指令?硬件是否需要读取8个字节,更改一个字节并存储它(使用2次内存操作)?
编辑:专门针对 x86-64 架构
在 x86-64 上,硬件将读取一个 缓存行 ,修改缓存中的字节,最终将缓存行写回内存。
发生回写的主要原因是CPU需要其他数据的缓存行。有强制回写的明确说明,但 C 编译器不太可能使用这些说明。它会减慢 CPU 以强制进行不必要的写入。
这完全取决于编译器、优化等。只需尝试编译和反汇编即可。作为示例,我们将编译以下代码:
#include <stdio.h>
int main() {
char a[] = "01234567890";
a[5] = 'A';
printf("%s\n", a);
}
// gcc -o main -std=c11 -Wall -Wextra -O0 main.c
我们通过objdump
得到反汇编:
6c1: 48 b8 30 31 32 33 34 movabs rax,0x3736353433323130
6c8: 35 36 37
6cb: 48 89 45 ec mov QWORD PTR [rbp-0x14],rax
6cf: c7 45 f4 38 39 30 00 mov DWORD PTR [rbp-0xc],0x303938
6d6: c6 45 f1 41 mov BYTE PTR [rbp-0xf],0x41
6da: 48 8d 45 ec lea rax,[rbp-0x14]
6de: 48 89 c7 mov rdi,rax
// objdump -d ./main -Mintel | less