为什么我的 C 程序中的虚拟地址如此之大?

Kro*_*szi 0 c virtual-memory

我最近了解了虚拟内存和分页,并且编译器只生成从 1 开始并简单向上计数的虚拟地址。我想我会测试这个并在下面写了一个简短的 C 程序,它实例化一个全局变量并打印它的地址,期望一个非常小的值,因为 CPU 只看到虚拟地址,但我得到4247584。这是怎么回事,我的假设是错误的吗?如果可能的话,显示从 1 开始生成虚拟地址的程序是什么?

我的程序:

#include <stdio.h>

int x = 0;

int main(){
    printf("%d\n", &x);
    return 0;
}
Run Code Online (Sandbox Code Playgroud)

(我在 Windows 10 上使用 gcc 4.8.1)

tof*_*fro 5

虚拟地址的实际相对来说并不重要(好吧,因为它是虚拟的)。当它不是从 0 开始时,没有任何“浪费”。地址值的唯一前提是程序、数据及其所有关联的共享库实际上适合值空间。

然而,出于安全原因,以潜在攻击者不可复制的方式在虚拟地址空间中分配进程的各种代码和数据区域是有意义的(使固定地址的代码注入攻击几乎不可能[原文如此]),即这就是现代操作系统为程序随机分配虚拟地址空间值的原因。

在某些操作系统(如 Linux)上,您可以关闭虚拟地址空间布局随机化,从而使其可重现。地址很可能仍然不会从零开始,因为库和启动代码很可能会占用比您自己的程序低的地址。