来自这个问题:函数指针:p hysical 或虚拟地址
我明白,当我打印任何变量的地址时,它打印的是虚拟地址。但我在书中读到,对于每个进程,虚拟地址从0开始,MMU实际上将负责地址转换。
那么当我多次执行程序时,看到一些随机地址?为什么会这样?为什么不从 0 开始?地址是否表示与实际进程的基址有一些偏移?如果是这样,不是意味着它是物理地址吗?
我在 POP 操作系统上运行我的电脑,我的系统也是新的。 以下是我在每次执行中打印变量地址时的值:
0x7ffc62f83904
0x7ffd78dee214
0x7fff2b69f6c4
0x7ffcc89680a4
0x7ffdc5cbf514
0x7ffd00540714
0x7ffd743ac3d4
0x7ffe7c5a9914
0x7ffd1ea1d214
0x7ffe21c30d64
0x7ffe885d9de4
0x7ffc5d8432e4
0x7ffe87ebff04
0x7ffc726e88d4
0x7ffef650e684
0x7fff0a62a2f4
0x7ffe89e3aed4
0x7ffd77e596f4
0x7ffcfcf76c54
0x7fff9a2cf654
0x7ffe3a8cbf84
0x7ffc4e127704
0x7ffce1a9f894
0x7ffd908828a4
0x7ffc88b0c6b4
0x7ffc2dd04804
0x7ffda3991c24
0x7fffad288da4
0x7ffd34da2994
0x7ffc3e3dcf54
0x7ffc6780b224
0x7ffee8f41554
0x7ffd369d9c54
0x7ffe5bdfca14
0x7ffc9a772454
0x7ffe03665d04
0x7ffce0eeb234
0x7ffc4fad04a4
0x7ffea8c715e4
0x7ffc6c3eb7f4
0x7ffea32a5e24
0x7ffea7729394
0x7ffdfbd2eab4
0x7fff36934134
0x7ffe9e15b1e4
0x7ffe12a07194
0x7ffc1b2fce34
0x7ffc3a82b684
0x7ffe56aad1d4
0x7ffd00d59c54
0x7fff58636474
0x7ffdc8940964
0x7ffed8146aa4
0x7ffd81794c94
0x7ffdce6ac874
0x7fff761a79c4
0x7ffe8fc95a24
0x7ffd3fb95464
0x7ffdbceeeb84
0x7ffc1582c2d4
0x7ffd47f0e8b4
0x7ffdd55b12f4
0x7ffc802a3db4
0x7ffebe9634a4
0x7ffec809acf4
0x7ffd0dfe0354
0x7ffc80eeb8d4
0x7fff1914b3b4
0x7fff3fee60d4
0x7ffccf1febf4
0x7ffdfe68c264
0x7ffd631c8184
0x7ffd814bc3c4
0x7ffc53a2d3d4
0x7fff49d96e14
0x7ffc16144e14
0x7ffecfe11904
0x7ffeaa3dc584
0x7ffcfb3a2894
0x7fff8a6a04b4
0x7ffcf0e035f4
0x7ffe060a7654
0x7ffce2a00954
0x7ffcc81980f4
0x7ffc0c706034
0x7fff172e4f34
0x7ffd62c963e4
0x7ffcf00501a4
0x7ffeec50b044
0x7ffd0bbb2be4
0x7ffff8fbebf4
0x7ffd6e127ac4
0x7fff0f2b9714
0x7ffe8b6da014
0x7ffc58f83344
0x7ffc3aa463e4
0x7fff8b28c9d4
0x7ffe1799f8a4
0x7fffe398f734
0x7fffb3056f74
它不应该包含重复项吗? 现在,如果我并行多次运行同一个程序,它会包含重复项,这将确认它是虚拟地址而不是物理地址吗?
不,虚拟地址不需要从零开始,通常也不这样做。每个操作系统、程序加载程序和/或链接器都有一些默认内存布局。通常,地址 0 处不映射任何内容,以便零可以充当空指针,并在取消引用时导致错误。通常,堆栈在内存中的位置略高,但通过随机偏移进行调整,以挫败依赖于固定或已知地址的攻击。这就是导致你看到的变化的原因。
那么每次编译时地址都应该相同(例如0x28fefc(。不,它不应该从 0 开始,因为 MMU 会进行转换。
使用此代码重试,看看它是否有效:
#include<iostream>
using namespace std;
int main(){
int n;
cout << &n'
}
要查看存储在地址中的值,请尝试以下操作,这称为取消引用:
#include<iostream>
using namespace std;
int main(){
int n = 10;
cout << *(&n);
}
如果问题仍然存在,我认为是新的操作系统。