有了這個神器,再也不怕shell寫得不對了
寫過shell腳本的人都知道,即便出現(xiàn)一些簡單的語法錯誤,運行的時候也可能沒有辦法發(fā)現(xiàn)。有些看似運行正確的腳本,實際上可能在某些分支,某些場景下仍然出現(xiàn)錯誤,而有的寫法可能運行正常,但是卻不符合POSIX標(biāo)準(zhǔn),不具備可移植性。
誠然,shell腳本是解釋運行,沒有辦法向C/C++那樣嚴(yán)格檢查,但是我們?nèi)匀豢梢越柚恍┕ぞ邘椭覀兲崆鞍l(fā)現(xiàn)一些錯誤。
shellcheck
shellcheck就是這樣的一個工具。它可以在多種場景下使用,包括在線,命令行檢查,編輯器配置,下面逐一介紹。
在線使用
顧名思義,它提供了一個在線的檢查地址,https://www.shellcheck.net/,進入網(wǎng)址即可使用。
例如,你輸入你的腳本內(nèi)容:
- #!/bin/sh
 - for n in {1..$RANDOM}
 - do
 - str=""
 - if (( n % 3 == 0 ))
 - then
 - str="fizz"
 - fi
 - if [ $[n%5] == 0 ]
 - then
 - str="$strbuzz"
 - fi
 - if [[ ! $str ]]
 - then
 - str="$n"
 - fi
 - echo "$str"
 - done
 
shell
它會給出錯誤提示或者建議:
- Line 2:
 - for n in {1..$RANDOM}
 - ^-- SC2039: In POSIX sh, brace expansion is undefined.
 - ^-- SC2039: In POSIX sh, RANDOM is undefined.
 - Line 5:
 - if (( n % 3 == 0 ))
 - ^-- SC2039: In POSIX sh, standalone ((..)) is undefined.
 - Line 9:
 - if [ $[n%5] == 0 ]
 - ^-- SC2039: In POSIX sh, $[..] in place of $((..)) is undefined.
 - ^-- SC2007: Use $((..)) instead of deprecated $[..]
 - ^-- SC2039: In POSIX sh, == in place of = is undefined.
 - Line 11:
 - str="$strbuzz"
 - ^-- SC2154: strbuzz is referenced but not assigned.
 - Line 13:
 - if [[ ! $str ]]
 - ^-- SC2039: In POSIX sh, [[ ]] is undefined.
 
怎么樣,是不是很給力,每個可能的錯誤都提示了。新手寫shell出現(xiàn)莫名的報錯時,可以嘗試使用奧。當(dāng)然例子中很多并不是真的錯誤,而是某種寫法不符合POSIX標(biāo)準(zhǔn),這種情況也應(yīng)該避免。
關(guān)于shell的基本內(nèi)容,也可以參考《shell必備基礎(chǔ)知識》。
命令行使用
命令行安裝也很簡單(記得使用root權(quán)限),ubuntu下:
- $ apt-get install shellcheck
 
centos下:
- $ yum -y install epel-release
 
Fedora下:
- $ dnf install ShellCheck
 
使用方法也很簡單了:
- $ shellcheck myscript.sh
 
舉個例子,下面的寫法是新手最容易出錯的地方之一:
- //來源:公眾號【編程珠璣】
 - //作者:守望先生
 - #!/bin/bash
 - if[ $# -eq 0 ]
 - then
 - echo "no para"
 - else
 - echo "$# para"
 - fi
 - exit 0
 
看運行報錯:
- ./test.sh: line 4: if[ 0 -eq 0 ]: command not found
 - ./test.sh: line 5: syntax error near unexpected token `then'
 - ./test.sh: line 5: `then'
 
只是告訴你在then附近有語法問題,到底什么問題呢?我們用shellcheck看看:
- $ shellcheck test.sh
 - In test.sh line 4:
 - if[ $# -eq 0 ]
 - ^-- SC1069: You need a space before the [.
 
這么一看,就很清楚了,原來[前面少了空格。
編輯器中使用
當(dāng)然也可以把它安裝到你熟悉的編輯器中,雖然它們本身都有語法高亮的功能,但是并沒有直接的信息提示,安裝shellcheck類工具,達到編寫即提示的效果。
- Emacs, 可以使用 Flycheck.
 - Sublime,可以使用 SublimeLinter.
 - Atom,可以使用 Linter.
 - vim ,可以使用ale或者syntastic
 
當(dāng)然了,現(xiàn)代很多IDE都有這樣檢查功能,這里只說編輯器。
這里以syntastic為例,實際上它支持多種語言的語法檢查。
安裝過程:
1.安裝pathogen.vim
- $ mkdir -p ~/.vim/autoload ~/.vim/bundle && \
 - curl -LSso ~/.vim/autoload/pathogen.vim https://tpo.pe/pathogen.vim
 
并且在vimrc文件中配置以下內(nèi)容:
- execute pathogen#infect()
 
2.安裝 Install syntastic
- cd ~/.vim/bundle && \
 - git clone --depth=1 https://github.com/vim-syntastic/syntastic.git
 
3.測試安裝情況
打開vim,輸入以下內(nèi)容
- :Helptags
 
如果沒有報錯,說明安裝正常。
在vimrc中配置以下內(nèi)容:
- set statusline+=%#warningmsg#
 - set statusline+=%{SyntasticStatuslineFlag()}
 - set statusline+=%*
 - let g:syntastic_always_populate_loc_list = 1
 - let g:syntastic_auto_loc_list = 1
 - let g:syntastic_check_on_open = 1
 
常用:
- :Errors 顯示錯誤面板
 - :lnext 到下一個錯誤
 - :lprevious 到上一個錯誤
 
更多安裝詳情也可以參考https://github.com/vim-syntastic/syntastic。
以上是官網(wǎng)推薦的安裝方式,也可以在安裝了Vundle(這是一種老舊的插件管理方式,你可以嘗試vim-plug等其他插件管理工具)的前提下,通過在配置文件中加入:
- Plugin 'scrooloose/syntastic'
 
打開vim輸入:
- :PluginInstall
 
即可安裝。
,具體安裝方式可以參考《vim完整開發(fā)環(huán)境配置--老舊版》。
使用效果:
shell檢查:
C語言語法檢查:
實際上它可以支持幾乎所有常見編程語言的語法檢查。
具體可以查看這里
https://github.com/vim-syntastic/syntastic/blob/master/doc/syntastic-checkers.txt
不知道vimrc文件在哪里?
打開vim,輸入:
- :version
 
就可以看到啦:
- system vimrc file: "$VIM/vimrc"
 - user vimrc file: "$HOME/.vimrc"
 - 2nd user vimrc file: "~/.vim/vimrc"
 - user exrc file: "$HOME/.exrc"
 - system gvimrc file: "$VIM/gvimrc"
 - user gvimrc file: "$HOME/.gvimrc"
 - 2nd user gvimrc file: "~/.vim/gvimrc"
 - system menu file: "$VIMRUNTIME/menu.vim"
 
它們區(qū)別在于生效范圍不一樣,對于用戶的vimrc,自然只是對特定用戶生效。
總結(jié)
工欲善其事必先利其器,有好的工具,自然就該用起來。歡迎分享更多的方法或工具。

















 
 
 











 
 
 
 