为什么执行同一个程序每次输出的变量地址是不一样的

首先看一下下面的代码

#include<stdio.h>

int main()
{
    int a = 1;
    printf("%p\n", &a);
    return 0;
}

在这里插入图片描述
然后我就很疑惑,为什么每次的地址都是不一样的。
为什么会有这样的疑问呢,编译完成后得到的文件叫做可重定位目标文件链接的过程就是主要就是地址空间分配,符号决议和重定位。那么,按照这样的逻辑推断,gcc test1.c后得到在这里插入图片描述可执行文件确定后,相应的变量的地址空间也就确定了,那么为什么输出的地址却不一样????
后来再和同学的讨论过程中得知了,地址空间随机化分布

在这里插入图片描述
切换到root权限后执行 echo 0 >/proc/sys/kernel/randomize_va_space
然后运行a.out文件,可以看出每次输出的结果都是一样的
实验结束,记得一定要 echo 2 >/proc/sys/kernel/randomize_va_space,打开地址空间随机化的开关,这是为了防止对抗缓冲区溢出的攻击。
实验环境: 4.15.0-29deepin-generic x86_64 GNU/Linux gcc version 7.3.0
提示:可能你试验的结果和我并不一致,因为地址空间随机化,需要操作系统和程序自身的双重支持
在这里,我并没有详细介绍地址空间随机化,一方面是我理解的有限,写一写也只是泛泛而谈;其次,我觉得这就是一篇科普性质的文章,能让你们知道有这个机制就可以了……

  • 11
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,我明白了。我将执行你的要求,但是我需要提醒你,这是一个经典的并发编程题,程序的结果可能会因为系统、硬件等因素而有所不同。 以下是程序C语言代码: ``` #include <stdio.h> #include <stdlib.h> #include <unistd.h> #define LOOP_TIMES 10000 int main() { int var = 0; pid_t pid; pid = fork(); if (pid == 0) { for (int i = 0; i < LOOP_TIMES; i++) { var--; printf("a[%d]a\n", var); } } else if (pid > 0) { for (int i = 0; i < LOOP_TIMES; i++) { var++; printf("A[%d]A\n", var); } } else { perror("fork"); exit(1); } return 0; } ``` 这段代码首先定义了一个整型变量 var,初始值为 0。然后使用 fork() 函数创建了一个子进程,在父子进程各自循环 10000 次,父进程不断地将 var 增加 1,然后输出 A[var 的值]A,子进程不断地将 var 减小 1,然后输出 a[var 的值]a。 由于父子进程是并发执行的,所以在执行过程会产生交替输出的现象。如果程序顺利运行,最终的输出结果应该是一系列交替输出的 A 和 a,且 A 和 a 的个数应该相等。 在重新运行程序并将结果重定向到 result.txt 文件后,可以使用文件查看器或者命令行工具查看 result.txt 的内容。运行结果应该与在屏幕上显示的结果是一样的,只是没有交替的输出效果。这是因为重定向将程序输出结果写入文件,并没有用于显示在屏幕上的终端设备,所以看不到交替输出的效果。 但是,运行结果可能会因为系统、硬件等因素而略有不同。在运行过程可能会发生竞态条件、饥饿、死锁等并发编程难题。因此,多次运行程序并观察结果可以得到更完整的结果集合,并且可以更好地了解程序在不同环境下的执行情况。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值