Linux環境變量和進程地址空間介紹

Linux環境變量和進程地址空間

//查看進程pid和父進程
[dy@VM-12-10-centos jincheng_12_5]$ ps ajx | head -1 && ps axj | grep 3669470
   PPID     PID    PGID     SID TTY        TPGID STAT   UID   TIME COMMAND
3669469 3669470 3669470 3669439 pts/0    3681645 S     1001   0:00 -bash
3669470 3681645 3681645 3669439 pts/0    3681645 R+    1001   0:00 ps axj
3669470 3681646 3681645 3669439 pts/0    3681645 R+    1001   0:00 grep --color=auto 3669470


//第1種           不需要加路徑就能運行程序的方法
[dy@VM-12-10-centos jincheng_12_5]$ ll
total 28
-rw-rw-r-- 1 dy dy    73 Dec  5 16:46 Makefile
-rwxrwxr-x 1 dy dy 17432 Dec  5 16:47 proc
-rw-rw-r-- 1 dy dy    75 Dec  5 16:47 proc.c
[dy@VM-12-10-centos jincheng_12_5]$ sudo cp proc /usr/bin/                  
//將自己的可執行程序添加到系統路徑下
[sudo] password for dy: 
[dy@VM-12-10-centos jincheng_12_5]$ proc //將自己的可執行程序添加到系統路徑下,就可以不要加./路徑瞭
hello world!
[dy@VM-12-10-centos jincheng_12_5]$ ./proc
hello world!
[dy@VM-12-10-centos jincheng_12_5]$ sudo rm /usr/bin/proc //從系統路徑下刪除
[dy@VM-12-10-centos jincheng_12_5]$ proc
-bash: /usr/bin/proc: No such file or directory

//第二種
[dy@VM-12-10-centos jincheng_12_5]$ $PATH
-bash: /home/dy/.local/bin:/home/dy/bin:/usr/local/bin:/usr/bin:/usr/local/sbin:/usr/sbin: No such file or directory
[dy@VM-12-10-centos jincheng_12_5]$ pwd
/home/dy/jincheng_12_5
//將當前路徑添加到環境變量PATH當中
[dy@VM-12-10-centos jincheng_12_5]$ PATH=$PATH:/home/dy/jincheng_12_5
[dy@VM-12-10-centos jincheng_12_5]$ proc
hello world!

//隻保存3000條命令
[dy@VM-12-10-centos jincheng_12_5]$ echo $HISTSIZE
3000
//顯示之前有多少條命令
[dy@VM-12-10-centos jincheng_12_5]$ history |wc -l
636
[dy@VM-12-10-centos jincheng_12_5]$ history
//將之前的命令全部輸出到顯示器上
   
//顯示所有環境變量
[dy@VM-12-10-centos jincheng_12_5]$ env
LS_COLORS=rs=0:di=01;34:ln=01;36:mh=00:pi=40;33:so=01;35:do=01;35:bd=40;33;01:cd=40;33;01:or=40;31;01:mi=01;05;37;41:su=37;41:sg=30;43:ca=30;41:tw=30;42:ow=34;42:st=37;44:ex=01;32:*.tar=01;31:*.tgz=01;31:*.arc=01;31:*.arj=01;31:*.taz=01;31:*.lha=01;31:*.lz4=01;31:*.lzh=01;31:*.lzma=01;31:*.tlz=01;31:*.txz=01;31:*.tzo=01;31:*.t7z=01;31:*.zip=01;31:*.z=01;31:*.dz=01;31:*.gz=01;31:*.lrz=01;31:*.lz=01;31:*.lzo=01;31:*.xz=01;31:*.zst=01;31:*.tzst=01;31:*.bz2=01;31:*.bz=01;31:*.tbz=01;31:*.tbz2=01;31:*.tz=01;31:*.deb=01;31:*.rpm=01;31:*.jar=01;31:*.war=01;31:*.ear=01;31:*.sar=01;31:*.rar=01;31:*.alz=01;31:*.ace=01;31:*.zoo=01;31:*.cpio=01;31:*.7z=01;31:*.rz=01;31:*.cab=01;31:*.wim=01;31:*.swm=01;31:*.dwm=01;31:*.esd=01;31:*.jpg=01;35:*.jpeg=01;35:*.mjpg=01;35:*.mjpeg=01;35:*.gif=01;35:*.bmp=01;35:*.pbm=01;35:*.pgm=01;35:*.ppm=01;35:*.tga=01;35:*.xbm=01;35:*.xpm=01;35:*.tif=01;35:*.tiff=01;35:*.png=01;35:*.svg=01;35:*.svgz=01;35:*.mng=01;35:*.pcx=01;35:*.mov=01;35:*.mpg=01;35:*.mpeg=01;35:*.m2v=01;35:*.mkv=01;35:*.webm=01;35:*.ogm=01;35:*.mp4=01;35:*.m4v=01;35:*.mp4v=01;35:*.vob=01;35:*.qt=01;35:*.nuv=01;35:*.wmv=01;35:*.asf=01;35:*.rm=01;35:*.rmvb=01;35:*.flc=01;35:*.avi=01;35:*.fli=01;35:*.flv=01;35:*.gl=01;35:*.dl=01;35:*.xcf=01;35:*.xwd=01;35:*.yuv=01;35:*.cgm=01;35:*.emf=01;35:*.ogv=01;35:*.ogx=01;35:*.aac=01;36:*.au=01;36:*.flac=01;36:*.m4a=01;36:*.mid=01;36:*.midi=01;36:*.mka=01;36:*.mp3=01;36:*.mpc=01;36:*.ogg=01;36:*.ra=01;36:*.wav=01;36:*.oga=01;36:*.opus=01;36:*.spx=01;36:*.xspf=01;36:
LANG=en_US.utf8
HISTTIMEFORMAT=%F %T 
HOSTNAME=VM-12-10-centos
OLDPWD=/home/dy
USER=dy
PWD=/home/dy/jincheng_12_5
HOME=/home/dy
MAIL=/var/spool/mail/dy
SHELL=/bin/bash
TERM=xterm
SHLVL=1
PROMPT_COMMAND=history -a; history -a; printf "\033]0;%s@%s:%s\007" "${USER}" "${HOSTNAME%%.*}" "${PWD/#$HOME/\~}"
LOGNAME=dy
PATH=/home/dy/.local/bin:/home/dy/bin:/usr/local/bin:/usr/bin:/usr/local/sbin:/usr/sbin:/home/dy/jincheng_12_5
HISTSIZE=3000
LESSOPEN=||/usr/bin/lesspipe.sh %s
_=/usr/bin/env


echo: 顯示某個環境變量值

export: 設置一個新的環境變量

[dy@VM-12-10-centos jincheng_12_5]$ ./proc  //本地變量沒有
I am a proc : pid:3700079 ppid:3669470
Segmentation fault (core dumped)
[dy@VM-12-10-centos jincheng_12_5]$ export MY_VAL
[dy@VM-12-10-centos jincheng_12_5]$ ./proc
I am a proc : pid:3700163 ppid:3669470
hello dy
/home/dy/.local/bin:/home/dy/bin:/usr/local/bin:/usr/bin:/usr/local/sbin:/usr/sbin:/home/dy/jincheng_12_5
/home/dy
/bin/bash

env: 顯示所有環境變量

unset: 清除環境變量

env 環境變量

set: 顯示本地定義的shell變量和環境變量

//設置環境變量
[dy@VM-12-10-centos jincheng_12_5]$ hello=1234
//輸出環境變量 記得要加$符
[dy@VM-12-10-centos jincheng_12_5]$ echo $hello
1234
[dy@VM-12-10-centos jincheng_12_5]$ set | grep hello
hello=1234

在命令行上運行的大部分的指令,它的父進程都是bash,bash創建子進程,子進程執行你的命令。

通過代碼獲取環境變量

#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
int main(int argc, char *argv[], char *env[])
{
        for(int i = 0; argv[i]; ++i)//為命令行參數做解釋
        //for(int i = 0; i < argc; ++i)
        {
            printf("argv[%d]:%s\n",i,argv[i]);
        }
        for(int i = 0; env[i]; ++i)//可以通過代碼獲取環境變量
        {
           printf("env[%d]:%s\n",i,env[i]);
        }
   return 0;
}

[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-7v2pxCt8-1639035758639)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\image-20211208141656121.png)]

int x = atoi(argv[2]);
int y = atoi(argv[3]);
if(strcmp(argv[1],"-a") == 0)
{
    printf("%d + %d = %d\n",x,y,x+y);
}
else
{
    printf("%d - %d = %d\n",x,y,x-y);
}

在這裡插入圖片描述

每個程序都會收到一張環境表,環境表是一個字符指針數組,每個指針指向一個以’\0’結尾的環境字符串

在這裡插入圖片描述

也可以通過第三方變量environ獲取環境

#include <stdio.h>
int main(int argc, char *argv[])
{
    extern char **environ;
    int i = 0;
    for(; environ[i]; i++){
    	printf("%s\n", environ[i]);
    }
	return 0;
}

在這裡插入圖片描述

進程地址空間

#include<stdio.h>
#include<unistd.h>
int g_val = 0;
int main()
{
   pid_t id = fork();
   if(id < 0)
   {
       perror("fork() fail\n");
   }
   else if(id == 0)
   {
       printf("I am a child:%d,%d,%p\n",getpid(),g_val,&g_val);
   }else
   {
       printf("I am a fatcher:%d,%d,%p\n",getppid(),g_val,&g_val);
   }
   return 0;
}

在這裡插入圖片描述

通過運行結果,我們看到兩個進程的變量的值和地址是一樣的,這裡沒有對代碼做任何更改,子進程按照父進程為模板,所以地址是一樣的。

我們稍微將代碼更改一下

#include<stdio.h>
#include<unistd.h>
int g_val;
int main()
{
   pid_t id = fork();
   if(id < 0)
   {
       perror("fork() fail");
   }
   else if(id == 0)
   {
       g_val = 100;
       printf("I am a child:%d,%d,%p\n",getpid(),g_val,&g_val);
   }else
   {
       sleep(3);
       printf("I am a fatcher:%d,%d,%p\n",getppid(),g_val,&g_val);
   }
   return 0;
}

在這裡插入圖片描述

此時我們發現,變量的值已經更改瞭,父進程為0,子進程為100,但是變量的地址仍然是一樣的。

通過這個運行結果,我們可以得知以下幾點。

兩個進程的變量的值不一樣,所以父子進程的變量絕對不是同一個變量地址值一樣,絕對不是物理地址!!!我們看到的地址,全部都是虛擬地址!!!真實的物理地址,用戶是看不到的,由OS統一管理(OS負責將虛擬地址–》物理地址)代碼是共享的,而數據各自私有一份(寫時拷貝)

關於為什麼是虛擬地址,而不是物理地址?

進程地址空間是什麼?

進程地址空間:是看待內存的方式,抽象出來的一個概念,內核struct mm struct,這樣的每個進程,都認為自己獨占系統內存資源(你認為你老爸的傢產是你一個人的)區域劃分:地址空間為線性,被劃分成一個一個的區域,[start,end]虛擬地址:在[start, end]之間的各個地址叫做虛擬地址

在這裡插入圖片描述

虛擬地址到物理地址的一個映射

在這裡插入圖片描述

為什麼要存在進程地址空間?

保護物理內存,不收到任何進程內的地址的直接訪問,方便進行合法性檢驗將內存管理和進程管理進行解耦讓每個進程,以同樣的方式,來看待代碼和數據

[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-wu0C4V57-1639035758649)(C:\Users\Administrator\AppData\Roaming\Typora\typora-user-images\image-20211209150456785.png)]

以後我們可以直接說出進程和程序的區別。

在這裡插入圖片描述

到此這篇關於Linux環境變量和進程地址空間介紹的文章就介紹到這瞭,更多相關Linux環境變量和進程空間內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: