2017.6.29
理解shell內(nèi)建命令
理解內(nèi)建命令和外部命令非常重要广恢。相比之下绞愚,外部命令使用成本更高他嫡。
外部命令
外部命令,是存在于bash shell之外的程序备籽,不是shell程序的一部分舶治。通常位于/bin, /usr/bin, /sbin, /usr/sbin中。
例如ps
胶台,我們來看看
wsx@wsx-ubuntu:~$ which ps
/bin/ps
wsx@wsx-ubuntu:~$ type -a ps
ps 是 /bin/ps
wsx@wsx-ubuntu:~$ ls -l /bin/ps
-rwxr-xr-x 1 root root 97408 11月 22 2016 /bin/ps
當(dāng)外部命令執(zhí)行時(shí)歼疮,會(huì)創(chuàng)建出一個(gè)子進(jìn)程。這種操作成為衍生(forking)诈唬。
內(nèi)建命令
內(nèi)建命令和外部命令的區(qū)別在于前者不需要使用子進(jìn)程來執(zhí)行韩脏。它們已經(jīng)和shell編譯成了一體,作為shell工具的組成部分存在铸磅。不需要借助外部程序文件來運(yùn)行赡矢。
例如cd
和exit
都內(nèi)建于bash shell≡淖校可以利用type
命令來了解某個(gè)命令是否是內(nèi)建的吹散。
wsx@wsx-ubuntu:~$ type cd
cd 是 shell 內(nèi)建
wsx@wsx-ubuntu:~$ type exit
exit 是 shell 內(nèi)建
有意思的是,有些命令有多種實(shí)現(xiàn)方式八酒,即它既有內(nèi)建空民,也有外部命令方式⌒呙裕可以通過type -a
查看界轩。
wsx@wsx-ubuntu:~$ type -a echo
echo 是 shell 內(nèi)建
echo 是 /bin/echo
wsx@wsx-ubuntu:~$ which echo
/bin/echo
wsx@wsx-ubuntu:~$ type -a pwd
pwd 是 shell 內(nèi)建
pwd 是 /bin/pwd
wsx@wsx-ubuntu:~$ which pwd
/bin/pwd
可以看到,type -a
命令顯示了每個(gè)命令的兩種實(shí)現(xiàn)衔瓮。而which
命令只顯示出了外部命令文件浊猾。
使用history命令
bash shell
會(huì)跟蹤你用過的命令,我們可以喚回這些命令并重新使用热鞍。通常記錄保存1000條葫慎,可以通過HISTSIZE
環(huán)境變量進(jìn)行修改。
通過!!
就可以重新喚出剛剛用過的命令
wsx@wsx-ubuntu:~$ !!
which pwd
/bin/pwd
當(dāng)我們使用多個(gè)終端會(huì)話進(jìn)行工作時(shí)薇宠,歷史記錄并不會(huì)自動(dòng)更新偷办,可以通過history -n
命令強(qiáng)制重新讀入。
環(huán)境變量
bash shell用一種叫做環(huán)境變量的特性來存儲(chǔ)有關(guān)shell會(huì)話和工作環(huán)境的信息昼接。這項(xiàng)特性允許你在內(nèi)存中存儲(chǔ)數(shù)據(jù)爽篷,以便程序或shell中運(yùn)行的腳本能夠輕松訪問到它們。
分為兩類:
- 全局變量
- 局部變量
全局環(huán)境變量
全局環(huán)境變量對(duì)于shell會(huì)話和所有生成子shell都是可見的慢睡。局部變量則只對(duì)創(chuàng)建它們的shell可見逐工。
Linux系統(tǒng)在你開始bash會(huì)話時(shí)就設(shè)置了一些全局環(huán)境變量铡溪。系統(tǒng)環(huán)境變量基本上都是使用全大寫字母,以區(qū)別于普通用戶的環(huán)境變量泪喊。
env
或printenv
命令可以查看全局變量:
wsx@wsx-ubuntu:~$ env
XDG_VTNR=7
XDG_SESSION_ID=c2
CLUTTER_IM_MODULE=xim
XDG_GREETER_DATA_DIR=/var/lib/lightdm-data/wsx
GPG_AGENT_INFO=/home/wsx/.gnupg/S.gpg-agent:0:1
TERM=xterm-256color
SHELL=/bin/bash
VTE_VERSION=4205
QT_LINUX_ACCESSIBILITY_ALWAYS_ON=1
WINDOWID=102760458
UPSTART_SESSION=unix:abstract=/com/ubuntu/upstart-session/1000/1819
GNOME_KEYRING_CONTROL=
GTK_MODULES=gail:atk-bridge:unity-gtk-module
JRE_HOME=/usr/lib/jvm/default-java/jre
......
環(huán)境變量很多棕硫,不大好找。如果想要顯示個(gè)別環(huán)境變量的值袒啼,可以用printenv
哈扮,不要用env
命令。
wsx@wsx-ubuntu:~$ printenv HOME
/home/wsx
可以用echo
命令顯示變量的值
wsx@wsx-ubuntu:~$ echo $HOME
/home/wsx
加上$
的變量還可以作為命令的參數(shù)
wsx@wsx-ubuntu:~$ ll $HOME
總用量 236220
drwxr-xr-x 65 wsx wsx 4096 6月 29 14:19 ./
drwxr-xr-x 3 root root 4096 9月 25 2016 ../
drwx------ 3 wsx wsx 4096 9月 25 2016 .adobe/
drwxrwxr-x 3 wsx wsx 4096 4月 13 10:59 .anaconda/
drwxrwxr-x 4 wsx wsx 4096 4月 14 15:05 .anydesk/
-rw------- 1 wsx wsx 54981 6月 29 10:50 .bash_history
-rw-r--r-- 1 wsx wsx 220 9月 25 2016 .bash_logout
-rw-r--r-- 1 wsx wsx 4203 6月 19 18:39 .bashrc
-rw-r--r-- 1 wsx wsx 4122 6月 19 18:39 .bashrc-anaconda3.bak
......
局部變量
顧名思義蚓再,局部環(huán)境變量只能在定義它們的進(jìn)程中可見滑肉。這兩點(diǎn)的區(qū)分相信接觸過編程的朋友都很容易理解,因?yàn)樵谝话憔幊陶Z言函數(shù)內(nèi)外摘仅,我們也必須涉及到這個(gè)概念的掌握靶庙。進(jìn)程或者子shell,我認(rèn)為都可以看做某種函數(shù)娃属。
事實(shí)上六荒,Linux系統(tǒng)也默認(rèn)定義了標(biāo)準(zhǔn)的局部環(huán)境變量。我們也可以自定義矾端。
頗為遺憾的是掏击,在Linux系統(tǒng)上并沒有一個(gè)只顯示局部環(huán)境變量的命令。set
命令會(huì)顯示為某個(gè)特定進(jìn)程設(shè)置的所有環(huán)境變量秩铆。
設(shè)置用戶自定義變量
設(shè)置局部用戶自定義變量
wsx@wsx-ubuntu:~$ echo $my_variable # 因?yàn)闆]定義砚亭,可以看到輸出為空
wsx@wsx-ubuntu:~$ my_variable=Hello # 等號(hào)兩邊不要留空格!
wsx@wsx-ubuntu:~$ echo $my_variable
Hello
wsx@wsx-ubuntu:~$ my_variable="Hello World" # 如果字符串還有空格殴玛,用引號(hào)括起來
wsx@wsx-ubuntu:~$ echo $my_variable
Hello World
wsx@wsx-ubuntu:~$ bash
wsx@wsx-ubuntu:~$ echo $my_variable # 在子shell中無法使用用戶自定義變量
設(shè)置全局環(huán)境變量
創(chuàng)建的方式是先創(chuàng)建一個(gè)局部環(huán)境變量钠惩,然后再把它導(dǎo)出到全局環(huán)境中。這個(gè)過程用export
命令來完成(但是退出終端后會(huì)失效族阅,需要保存到配置文件才能永久生效)。
刪除環(huán)境變量
用unset
可以完成這個(gè)操作膝捞,記住不要用$
坦刀。
記住一點(diǎn):只要用到變量,使用$
蔬咬;如果要操作變量鲤遥,不使用$
。一個(gè)例外是使用printenv
顯示某個(gè)變量的值林艘。
設(shè)置PATH環(huán)境變量
當(dāng)我們?cè)趕hell命令行界面輸入一個(gè)外部命令時(shí)盖奈,shell
必須搜索系統(tǒng)來找到對(duì)應(yīng)的程序。PATH
環(huán)境變量定義了用于進(jìn)行命令和程序?qū)ふ业哪夸洝?/p>
wsx@wsx-ubuntu:~$ echo $PATH
/home/wsx/Anaconda/bin:/home/wsx/Anoconda/ENTER/bin:/usr/lib/jvm/default-java/bin:/home/wsx/bin:/home/wsx/Anaconda/bin:/usr/local/sbin:/usr/local/bin:/usr/sbin:/usr/bin:/sbin:/bin:/usr/games:/usr/local/games:/snap/bin
目錄使用冒號(hào)分隔狐援。
可以添加搜索目錄到現(xiàn)有的PATH
環(huán)境變量钢坦,無需從頭定義究孕。你只需引用原來的PATH
值,然后再給這個(gè)字符串添加新目錄就行了爹凹。
$ PATH=$PATH:/home/scripts
$ echo $PATH
程序員通常的辦法是將單點(diǎn)符也加入PATH環(huán)境變量厨诸。該單點(diǎn)符代表當(dāng)前目錄。
$ PATH=$PATH:.
要記住禾酱,由父shell設(shè)置但并未導(dǎo)出的變量都是局部變量微酬,子shell無法繼承局部變量。
數(shù)組變量
要給某個(gè)環(huán)境變量設(shè)置多個(gè)值颤陶,可以把值放在括號(hào)里颗管,值與值之間用空格分隔。
wsx@wsx-ubuntu:~$ mytest=(one two three four five)
wsx@wsx-ubuntu:~$ echo $mytest
one
wsx@wsx-ubuntu:~$ echo ${mytest[2]} # 索引值從0開始
three
wsx@wsx-ubuntu:~$ echo ${mytest[*]}
one two three four five
wsx@wsx-ubuntu:~$ mytest[2]=seven # 改變某個(gè)索引值位置的值
wsx@wsx-ubuntu:~$ echo ${mytest[*]}
one two seven four five
wsx@wsx-ubuntu:~$ unset mytest[2] # 刪除某個(gè)變量
wsx@wsx-ubuntu:~$ echo ${mytest[*]}
one two four five