Linux:終端提示符 (prompt) 不如期生效原因
前言
先來簡單介紹下, prompt是什么鬼? 顧名思義就是提示符的意思, 看起來和我們遙遠, 但實際上只要是每個接觸shell的童鞋, 都有看到, 那就是我們在輸命令時前面的那串提示符.
例如:
當然, 這個樣式是可以修改的, 這就涉及到我們的PS1和PS2了, 有經驗或者以前有設置過的童鞋估計都不會陌生, 木有接觸過的童鞋可以參考一下鏈接學習下: linux PS1 提示符定義
問題
因為上面的并不是這次的主題, 所以簡單跳過. 其實我們的prompt除了能夠展示這些信息之外, 還能夠展示些較為'動態'的東西.
我們先來看個需求(簡單):
想要在每次執行命令時, 看到當前的時間
所以我們可以在~/.bash_profile里面加入下面的內容:
- export PS1="$(date +"%Y-%m-%d %H:%M:%S")\$ "
如代碼所示, 應該能夠在提示符上看到我們的時間, 現在試驗下:
正如我們猜測的一樣, 提示符那邊已經變成了我們的時間了, 但是貌似出現了一個問題, 當我們在執行下一條命令時, 時間卻沒有改變, 還是40秒, 哪怕我們多執行幾次還是這樣.
難道PS1只能加載一次? 但是這個原因很快就被否決, 因為當我們在切換用戶時, 提示符的$會改變成#, 而且也有其他的例子(下面會舉出來)證明, PS1是每次都會執行的. 所以問題只能在于我們寫法中.
考慮到是不是寫死命令的緣故, 于是就把date命令放到函數中, 看能否成功:
修改~/.bash_profile文件如下:
- echo_time(){
- date +"%Y-%m-%d %H:%M:%S"
- }
- export PS1="$(echo_time)\$ "
結果是失敗的, 貌似這樣的用法也是不行的! 查閱了很多文檔, 直到看到一篇文章說, 兩邊必須是單引號, 才恍然大悟, 開始有點理解為什么我的效果會失敗.
可能這里會有童鞋不清楚linux的單引號和雙引號的區別, 簡單來說就是:
雙引號: 讓大部分的符號(例如*), 失去意義,變為普通的字符.
單引號: 讓所有的符號, 都失去意義, 變為普通的字符.
舉個簡單的例子, echo "$a", 可以得出變量a的值, 但是echo '$a', 就真的只能輸出$a. 具體的細節, 可以查閱<shell十三問>.
原因分析
所以我這邊失敗的原因就是, 我這里需要用的是單引號, 而不是雙引號, 因為如果用雙引號, PS1在賦值時, 就已經獲得命令/函數的值了, 所以每次打印PS1, 都已經是具體的值了, 而不是一個命令/函數. 就好像:
- # 雙引號:
- PS1="$(date +%F)" 取出來的結果就是: PS1="2017-07-24"
- # 單引號:
- PS1='$(date +%F)' 取出來的結果就是: PS1='$(date +%F)'
所以等到在終端輸出時, 就會有不同的表現: 單引號, 每次輸出, 都是執行那個函數, 而雙引號, 因為已經是固定字符了, 所以每次都是那個值!
解決方案
找到了原因, 修改起來就得心應手了, 我們只需要將剛才的語法, 從雙引號換成單引號即可.
修改~/.bash_profile
- echo_time(){
- date +"%Y-%m-%d %H:%M:%S"
- }
- export PS1='$(echo_time)\$ '
可以看到這里已經是***解決了我們的問題了~
PS: 附上一個小小的福利
需求:
如果長期在終端下開發, 而且又是gitlab的專用戶的童鞋, 很多時候都會擔心自己開發的代碼分支會搞錯, 或者不記得自己所處的分支是啥, 經常就會需要git branch來看下, 現在我們可以通過這個, 來解決我們這個難題~
修改~/.bash_profile
- # 獲取代碼的分支
- get_branch(){
- BRANCH=`git branch 2>/dev/null |sed -n '/^\*/s/^\* //p'`
- if [[ -n $BRANCH ]]
- then
- echo " [*$BRANCH]"
- else
- echo ''
- fi
- }
- export PS1='\[\033[01;32m\]\u@\h\[\033[00m\]:\[\033[01;36m\]\W\[\033[00m\]\[\033[01;32m\]$(get_branch)\[\033[00m\] \$ '
運行的效果:
可以看到, 當我進入一個代碼目錄時, 我的提示符就會展示出所處的分支~, 這功能雖小, 但我覺得還是挺有用的, 就跟git bash客戶端類似, 希望大家喜歡!