今天我们继续学习进程相关知识。
PATH : 指定命令的搜索路径
HOME : 指定用户的主工作目录(即用户登陆到Linux系统中时,默认的目录)
SHELL : 当前Shell,它的值通常是/bin/bash。
为什么有些指令可以直接执行,不需要带路径,而我们的二进制程序需要带路径才能执行?
也就是说,如果我们把自己程序所在路径加入环境变量PATH当中,就可以直接执行。
测试:
[wkj@VM-4-13-centos ~]$ export PATH=$PATH:/home/wkj/lesson8
[wkj@VM-4-13-centos lesson8]$ ll
total 36
-rw-rw-r-- 1 wkj wkj 77 Nov 7 18:22 makefile
-rwxrwxr-x 1 wkj wkj 8464 Nov 12 21:23 myproc
-rw-rw-r-- 1 wkj wkj 169 Nov 12 21:23 myproc.c
-rwxrwxr-x 1 wkj wkj 9616 Nov 12 19:44 mytest
-rw-rw-r-- 1 wkj wkj 243 Nov 12 19:43 mytest.c
[wkj@VM-4-13-centos lesson8]$ myproc
hello world,20905
hello world,20905
hello world,20905
注意看,添加以后不需要./即可直接执行。
添加方法只在当前有用,退出登陆后不会保存,因此大家可以添加以下玩一玩,不用担心污染环境变量库。
#include
#include
int main()
{printf("%s\n", getenv("PATH"));return 0;
}
常用getenv和putenv函数来访问特定的环境变量。
tips:环境变量通常具有全局属性,可以被子进程继承下去
我们经常在各个论坛看到类似以下的图片。
#include
#include
#include
int g_val = 0;
int main()
{pid_t id = fork();if(id < 0){perror("fork");return 0;
}
else if(id == 0){ //childprintf("child[%d]: %d : %p\n", getpid(), g_val, &g_val);
}else{ //parentprintf("parent[%d]: %d : %p\n", getpid(), g_val, &g_val);
}sleep(1);return 0;
}
与环境相关,我们观察现象即可
parent[2995]: 0 : 0x80497d8
child[2996]: 0 : 0x80497d8
我们发现,输出出来的变量值和地址是一模一样的,很好理解呀,因为子进程按照父进程为模版,父子并没有对变量进行任何修改。可是将代码稍加改动:
#include
#include
#include
int g_val = 0;
int main()
{pid_t id = fork();if(id < 0){perror("fork");return 0;
}
else if(id == 0){ //child,子进程肯定先跑完,也就是子进程先修改,完成之后,父进程再读取g_val=100;printf("child[%d]: %d : %p\n", getpid(), g_val, &g_val);
}else{ //parentsleep(3);printf("parent[%d]: %d : %p\n", getpid(), g_val, &g_val);
}sleep(1);return 0;
}
与环境相关,我们观察现象即可
child[3046]: 100 : 0x80497e8
parent[3045]: 0 : 0x80497e8
怎么回事呢?地址一样,变量内容不一样,我接受不了哇!
我们可以得出以下结论:
页表就是一个用于将虚拟地址转换为物理地址的工具。在 riscv32 中,如果 satp 的第 31 位为 1 ,则表示启用页表机制。在访问虚拟内存时(开启页表时只允许访问虚拟内存),内存管理单元 MMU(Memory Management Unit) 会用过页表将虚拟地址转换为物理地址,再进行访问。
一个变量为什么可以标识不同的值?
父子进程各自在物理内存中,都有属于自己的变量空间,只不过在用户层用同一个变量(虚拟地址)来标识了。
以上就是今天要讲的内容啦。