shell指令碼入門
—、 shell入門簡介
1.1什麼是shell
- shell指令碼簡介
# 為啥介紹shell 上次出了一篇linux 命令詳解,得到了很多小夥伴的認可,有部分粉絲私信我,讓我出一份shell 程式設計。進過一段時間準備,花了2周時間,整理了一篇shell 入門到實戰的博文,歡迎大家閱讀,指點。 # 什麼是shell 網上有很多shell 的概念介紹,其實都很官方化,如果你對linux 命令很熟悉,那麼編寫shell 就不是一個難事,shell 本質上是 linux 命令,一條一條命令組合在一起,實現某一個目的,就變成了shell指令碼。它從一定程度上 減輕了工作量,提高了工作效率。 # 官方化的shell 介紹 Shell 通過提示您輸入,向作業系統解釋該輸入,然後處理來自作業系統的任何結果輸出,簡單來說Shell就是一個使用者跟作業系統之間的一個命令直譯器。 # 常見的shell 有哪些 Bourne Shell(/usr/bin/sh或/bin/sh) Bourne Again Shell(/bin/bash) C Shell(/usr/bin/csh) K Shell(/usr/bin/ksh) Shell for Root(/sbin/sh) # 最常用的shell是Bash,也就是Bourne Again Shell。Bash由於易用和免費,在日常工作中被廣泛使用,也是大多數Linux作業系統預設的Shell環境。
1.2 shell程式設計注意事項
- shell程式設計有哪些注意事項
shell命名: Shell指令碼名稱命名一般為英文、大寫、小寫,字尾以.sh 結尾
不能使用特殊符號、空格
見聞之意,名稱要寫的一眼可以看出功能. shell程式設計首行需要#I/bin/bash開頭
shell指令碼變數不能以數字、特殊符號開頭,可以使用下劃線—,但不能用破折號-
建立一個偉大程式設計專案—Hello World
1.3第一個shell指令碼 hello world
- 建立一個偉大程式設計專案—Hello World
# 建立一個Helloword.sh 檔案 [root@aly_server01~]# touch Helloword.sh # 編輯Helloword.sh 檔案 [root@aly_server01~]# vim Helloword.sh [root@aly_server01~]# cat Helloword.sh #!/bin/bash # This is ower first shell # by author rivers 2021.09 echo "hello world" [root@aly_server01~]# [root@aly_server01~]# ll Helloword.sh -rw-r--r-- 1 root root 85 Sep 20 22:26 Helloword.sh # 賦予執行許可權 [root@aly_server01~]# chmod o x Helloword.sh # 執行helloword.sh 指令碼 [root@aly_server01~]# ./Helloword.sh hello world [root@aly_server01~]#
二、shell環境變數講解
2.1 shell變數詳解
- 環境變數介紹
# 什麼是變數 很多人可能會說,可以變化的量就是變數。但是發現很多漢語意思很強大,你看的懂的字,卻不一定可以理解它的意思。這裡你可以理解為 a = 1,同時還可以 a =2、a = 3 ,不同的值都可以複製給同一個 變數 a 。 # 常見的3種變數 Shell程式設計中變數分為三種,分別是系統變數、環境變數和使用者變數,Shell變數名在定義時,首個字元必須為字母(a-z,A-Z),不能以數字開頭,中間不能有空格,可以使用下劃線(_),不能使用(-),也不能使用標點符號等。 # 簡單的變數介紹 [root@keeplived_server~]# a=18 [root@keeplived_server~]# echo $a 18
2.2 shell系統變數介紹
- 系統變數
# Shell常見的變數之一系統變數,主要是用於對引數判斷和命令返回值判斷時使用,系統變數詳解如下: $0 當前指令碼的名稱; $n 當前指令碼的第n個引數,n=1,2,…9; $* 當前指令碼的所有引數(不包括程式本身); $# 當前指令碼的引數個數(不包括程式本身); $? 令或程式執行完後的狀態,返回0表示執行成功; $$ 程式本身的PID號。
2.3 shell環境變數介紹
2.3.1常見的系統環境變數
- 環境變數介紹
#Shell常見的變數之二環境變數,主要是在程式執行時需要設定,環境變數詳解如下: PATH 命令所示路徑,以冒號為分割; HOME 列印使用者家目錄; SHELL 顯示當前Shell型別; USER 列印當前使用者名稱; ID 列印當前使用者id資訊; PWD 顯示當前所在路徑; TERM 列印當前終端型別; HOSTNAME 顯示當前主機名; PS1 定義主機命令提示符的; HISTSIZE 歷史命令大小,可通過 HISTTIMEFORMAT 變數設定命令執行時間; RANDOM 隨機生成一個 0 至 32767 的整數; HOSTNAME 主機名
2.4 shell 使用者環境變數介紹
2.4.1自定義shell環境變數
- 使用者自定義變數
# 常見的變數之三使用者變數,使用者變數又稱為區域性變數,主要用在Shell指令碼內部或者臨時區域性使用,系統變數詳解如下: a=rivers 自定義變數A; Httpd_sort=httpd-2.4.6-97.tar 自定義變數N_SOFT; BACK_DIR=/data/backup/ 自定義變數BACK_DIR; IPaddress=10.0.0.1 自定義變數IP1;
2.4.2 echo列印選單欄
- 使用echo列印選單欄,顯示http-2.4安裝過程
# echo 列印httpd-2.4安裝步驟 [root@web-server01~]# touch httpd_2.4_install.sh # 賦予執行許可權 [root@web-server01~]# chmod o x httpd_2.4_install.sh [root@web-server01~]# ./httpd_2.4_install.sh
2.4.3 shell中彩色輸出helloworld
- echo -e擴充套件
#!/bin/bash # This is echo color shell # by author rivers 2021.09-23 # 字型顏色 for i in {31..37}; do echo -e "/033[$i;40mHello world!/033[0m" done # 背景顏色 for i in {41..47}; do echo -e "/033[47;${i}mHello world!/033[0m" done # 顯示方式 for i in {1..8}; do echo -e "/033[$i;31;40mHello world!/033[0m" done
三、shell程式設計流程控制語句
3.1 if 條件語句介紹
3.1.1常用的單/雙分支
- if 條件語句
# If條件判斷語句,通常以if開頭,fi結尾。也可加入else或者elif進行多條件的判斷 # 單分支語句 ---比較大小 if (條件表示式);then 語句1 fi # 雙分支if 語句 if (表示式) 語句1 else 語句2 fi # 多支條件語句 ---判斷成績 if (表示式) 語句1 elif 語句2 elif 語句2 fi
3.1.2 if常見判斷邏輯運算子詳解
- if常見判斷邏輯運算子
-f 判斷檔案是否存在 eg: if [ -f filename ]; -d 判斷目錄是否存在 eg: if [ -d dir ]; -eq 等於,應用於整型比較 equal; -ne 不等於,應用於整型比較 not equal; -lt 小於,應用於整型比較 letter; -gt 大於,應用於整型比較 greater; -le 小於或等於,應用於整型比較; -ge 大於或等於,應用於整型比較; -a 雙方都成立(and) 邏輯表示式 –a 邏輯表示式; -o 單方成立(or) 邏輯表示式 –o 邏輯表示式; -z 空字串; -x 是否具有可執行許可權 || 單方成立; && 雙方都成立表示式。
3.1.3使用單分支語句判斷crond程序是否在執行---案例
- 判斷crond服務是否執行
#!/bin/bash # this is check crond # by author rivers on 2021-9.23 # 定義一個變數名 name=crond num=$(ps -ef|grep $name|grep -vc grep) if [ $num -eq 1 ];then echo "$num running!" else echo "$num is not running!" fi
3.1.4判斷系統目錄是否存在---案例
- 判斷系統目錄是否存在
#!/bin/bash # this is check directory # by author rivers on 2021-9.27 if [ ! -d /data/rivers -a ! -d /tmp/rivers ];then mkdir -p /data/rivers f i
3.1.5多個條件判斷學生分數等級---案例
- 判斷學生成績等級
# if 語句可以直接對命令狀態進行判斷,就省去了獲取$?這一步! # 如果第一個條件符合就不再向下匹配 #!/bin/bash # this check grade shell # by author rivers on 2021-09-27 grade=$1 if [ $grade -gt 90 ];then echo "Is's very good!" elif [ $grade -gt 70 ];then echo "Is's is good!" elif [ $grade -ge 60 ];then echo "pass" else echo "no pass" fi
3.2 for迴圈語句介紹
- for迴圈語句
#格式:for name [ [ in [ word ... ] ] ; ] do list ; done for 變數名 in 取值列表; do 語句 1 done
3.2.1檢查同—區域網多臺主機是否存活
- 檢查多臺主機存活情況
#!/bin/bash # check hosts is on/Off # by rivers on 20219-23 Network=$1 for Host in $(seq 1 254) do ping -c 1 $Network.$Host > /dev/null && result=0 || result=1 if [ "$result" == 0 ];then echo -e "/033[32;1m$Network.$Host is up /033[0m" echo "$Network.$Host" >> /tmp/up.txt else echo -e "/033[;31m$Network.$Host is down /033[0m" echo "$Network.$Host" >> /tmp/down.txt fi done
3.3 while迴圈語句介紹
- while迴圈語句
# While迴圈語句與for迴圈功能類似,主要用於對某個資料域進行迴圈讀取、對檔案進行遍歷,通常用於需要迴圈某個檔案或者列表,滿足迴圈條件會一直迴圈,不滿足則退出迴圈,其語法格式以while…do開頭,done結尾與 #while 關聯的還有一個 until 語句,它與 while 不同之處在於,是當條件表示式為 false 時才迴圈,實際使用中比較少,這裡不再講解。 while (表示式) do 語句1 done
- break和continue語句
# break 和 continue 語句 break 是終止迴圈。 continue 是跳出當前迴圈。 #示例 1:在死迴圈中,滿足條件終止迴圈 while true; do let N if [ $N -eq 5 ]; then break fi echo $N done 輸出: 1 2 3 4 #示例 2:舉例子說明 continue 用法 N=0 while [ $N -lt 5 ]; do let N if [ $N -eq 3 ]; then continue fi echo $N done 輸出: 1 2 4 # 列印 1-100 數字 i=0 while ((i<=100)) do echo $i i=`expr $i 1` done
3.3.1 While迴圈求1-100的總和---案例
- 求1-100的總和
#!/bin/bash # by author rivers on 2021-9-27 j=0 i=1 while ((i<=100)) do j=`expr $i $j` ((i )) done echo $j
3.3.2每10秒迴圈判斷—次hbs使用者是否登入系統---案例
- 每10秒迴圈判斷系統登入
[root@web-server01~/script]# vim login.sh #!/bin/bash #Check File to change. #By author rivers 2021-9-27 USERS="hbs" while true do echo "The Time is `date %F-%T`" sleep 10 NUM=`who|grep "$USERS"|wc -l` if [[ $NUM -ge 1 ]];then echo "The $USERS is login in system." fi done
3.4 case選擇語句介紹
- case選擇語句
#Case選擇語句,主要用於對多個選擇條件進行匹配輸出,與if elif語句結構類似,通常用於指令碼傳遞輸入引數,列印出輸出結果及內容,其語法格式以Case…in開頭,esac結尾。語法格式如下: case 模式名 in 模式 1) 命令 ;; 模式 2) 命令 ;; *) 不符合以上模式執行的命令 esac # 每個模式必須以右括號結束,命令結尾以雙分號結束。
3.4.1使用case編寫—個httpd服務啟動指令碼
- 編寫httpd服務啟動指令碼
[root@web-server01~/script]# vim httpd_start.sh # check http server start|stop|starus # by author rivers on 2021-9-27 while true do echo -e " /033[31m start /033[0m /033[32m stop /033[0m /033[33m status /033[0m /033[34m quit /033[0m " read -p "請輸入你的選擇start|stop|quit:" char case $char in start) systemctl start httpd && echo "httpd服務已經開啟" || echo "開啟失敗" ;; stop) systemctl stop httpd && echo "httpd服務已經關閉" || echo "關閉失敗" ;; restart) systemctl restart httpd && echo "httpd服務已經重啟" || echo "重啟失敗 " ;; status) systemctl status httpd && echo -e " httpd 的服務狀態 ;; quit)
3.5 select選擇語句介紹
- select選擇語句
#select 是一個類似於 for 迴圈的語句 #Select語句一般用於選擇,常用於選擇選單的建立,可以配合PS3來做列印選單的輸出資訊,其語法格式以select…in do開頭,done結尾: select i in (表示式) do 語句 done # 選擇mysql 版本 #!/bin/bash # by author rivers on 2021-9-27 PS3="Select a number: " while true; do select mysql_version in 5.1 5.6 quit; do case $mysql_version in 5.1) echo "mysql 5.1" break ;; 5.6) echo "mysql 5.6" break ;; quit) exit ;; *) echo "Input error, Please enter again!" break esac done done
3.5.1使用select列印lnmp選單欄---案例
- 列印lnmp選單欄
#!/bin/bash #by author rivers on 2021-9-27 PS3="Please enter you select install menu:" select i in http php mysql quit do case $i in http) echo -e " /033[31m Test Httpd /033[0m" ;; php) echo -e "/033[32m Test PHP/033[0m" ;; mysql) echo -e "/033[33m Test MySQL./033[0m" ;; quit) echo -e "/033[32m The System exit./033[0m" exit esac done
3.6 shell函式、陣列程式設計實戰
- 函式
# Shell允許將一組命令集或語句形成一個可用塊,這些塊稱為Shell函式,Shell函式的用於在於只需定義一次,後期隨時使用即可,無需在Shell指令碼中新增重複的語句塊,其語法格式以function name(){開頭,以}結尾。 # Shell程式設計函式預設不能將引數傳入()內部,Shell函式引數傳遞在呼叫函式名稱傳遞,例如name args1 args2。 # 函式語法 func() { command1 command1 …… } fun # 直接呼叫函式名 # Shell 函式很簡單,函式名後跟雙括號,再跟雙大括號。通過函式名直接呼叫,不加小括號。 #!/bin/bash func() { VAR=$((1 1)) return $VAR echo "This is a function." } func echo $? # bash test.sh 2
- 陣列
# 陣列是相同型別的元素按一定順序排列的集合。 格式:array=(元素 1 元素 2 元素 3 ...) 用小括號初始化陣列,元素之間用空格分隔。 定義方法 1:初始化陣列 array=(a b c) 定義方法 2:新建陣列並新增元素 array[下標]=元素 定義方法 3:將命令輸出作為陣列元素array=($(command))
3.6.1定義一個httpd安裝的函式---案例
- 建立apache軟體安裝函式
[root@web-server01~/script]# vim xx.sh #!/bin/bash #auto install apache #By author rivers 2021-09-27 #Httpd define path variable FILES=httpd-2.2.31.tar.bz2 LES_DIR=httpd-2.2.31 URL=http://mirrors.cnnic.cn/apache/httpd/ PREFIX=/usr/local/apache2/ function Apache_install () { #Install httpd web server if [[ "$1" -eq "1" ]];then wget -c $URL/$FILES && tar -jxvf $FILES && cd $FILES_DIR &&./configure if [ $? -eq 0 ];then make && make install echo -e "/n/033[32m-------------------------------------------- echo -e "/033[32mThe $FILES_DIR Server Install Success !/033[0m else echo -e "/033[32mThe $FILES_DIR Make or Make install ERROR,Plea exit 0 fi fi } Apache_install 1 # 呼叫函式,傳參為1
3.6.2遍歷陣列元素---案例
- 遍歷陣列元素
#方法 1: #!/bin/bash IP=(10.0.0.1 10.0.0.2 10.0.0.3) for ((i=0;i<${#IP[*]};i )); do echo ${IP[$i]} done # bash test.sh 10.0.0.1 10.0.0.2 10.0.0.3 #方法 2: #!/bin/bash IP=(10.0.0.1 10.0.0.2 10.0.0.3) for IP in ${IP[*]}; do echo $IP done
四、shell程式設計實戰案例
4.1 shell指令碼實戰之系統備份指令碼---案例
- Tar工具全備、增量備份網站,Shell指令碼實現自動打包備份
#!/bin/bash #Auto Backup Linux System Files #by author rivers on 2021-09-28 SOURCE_DIR=( $* ) TARGET_DIR=/data/backup/ YEAR=`date %Y` MONTH=`date %m` DAY=`date %d` WEEK=`date %u` A_NAME=`date %H%M` FILES=system_backup.tgz CODE=$? if [ -z "$*" ];then echo -e "/033[32mUsage:/nPlease Enter Your Backup Files or Directories/n--------------------------------------------/n/nUsage: { $0 /boot /etc}/033[0m" exit fi #Determine Whether the Target Directory Exists if [ ! -d $TARGET_DIR/$YEAR/$MONTH/$DAY ];then mkdir -p $TARGET_DIR/$YEAR/$MONTH/$DAY echo -e "/033[32mThe $TARGET_DIR Created Successfully !/033[0m" fi #EXEC Full_Backup Function Command Full_Backup() { if [ "$WEEK" -eq "7" ];then rm -rf $TARGET_DIR/snapshot cd $TARGET_DIR/$YEAR/$MONTH/$DAY ;tar -g $TARGET_DIR/snapshot -czvf $FILES ${SOURCE_DIR[@]} [ "$CODE" == "0" ]&&echo -e "--------------------------------------------/n/033[32mThese Full_Backup System Files Backup Successfully !/033[0m" fi } #Perform incremental BACKUP Function Command Add_Backup() { if [ $WEEK -ne "7" ];then cd $TARGET_DIR/$YEAR/$MONTH/$DAY ;tar -g $TARGET_DIR/snapshot -czvf $A_NAME$FILES ${SOURCE_DIR[@]} [ "$CODE" == "0" ]&&echo -e "-----------------------------------------/n/033[32mThese Add_Backup System Files $TARGET_DIR/$YEAR/$MONTH/$DAY/${YEAR}_$A_NAME$FILES Backup Successfully !/033[0m" fi } sleep 3 Full_Backup;Add_Backup
4.2 shell指令碼實戰之收集系統資訊、---案例
- shell指令碼實現伺服器資訊自動收集
cat <<EOF Welcome to use system Collect EOF ip_info=`ifconfig |grep "Bcast"|tail -1 |awk '{print $2}'|cut -d: -f 2` cpu_info1=`cat /proc/cpuinfo |grep 'model name'|tail -1 |awk -F: '{print $2}'|sed 's/^ //g'|awk '{print $1,$3,$4,$NF}'` cpu_info2=`cat /proc/cpuinfo |grep "physical id"|sort |uniq -c|wc -l` serv_info=`hostname |tail -1` disk_info=`fdisk -l|grep "Disk"|grep -v "identifier"|awk '{print $2,$3,$4}'|sed 's/,//g'` mem_info=`free -m |grep "Mem"|awk '{print "Total",$1,$2"M"}'` load_info=`uptime |awk '{print "Current Load: "$(NF-2)}'|sed 's//,//g'` mark_info='BeiJing_IDC' echo -e "/033[32m-------------------------------------------/033[1m" echo IPADDR:${ip_info} echo HOSTNAME:$serv_info echo CPU_INFO:${cpu_info1} X${cpu_info2} echo DISK_INFO:$disk_info echo MEM_INFO:$mem_info echo LOAD_INFO:$load_info echo -e "/033[32m-------------------------------------------/033[0m" echo -e -n "/033[36mYou want to write the data to the databases? /033[1m" ;read ensure if [ "$ensure" == "yes" -o "$ensure" == "y" -o "$ensure" == "Y" ];then echo "--------------------------------------------" echo -e '/033[31mmysql -uaudit -p123456 -D audit -e ''' "insert into audit_system values('','${ip_info}','$serv_info','${ cpu_info1} X${cpu_info2}','$disk_info','$mem_info','$load_info','$mark_info')" ''' /033[0m ' mysql -uroot -p123456 -D test -e "insert into audit_system values('','${ip_info}','$serv_info','${cpu_info1} X${cpu_info2} ','$disk_info','$mem_info','$load_info','$mark_info')" else echo "Please wait,exit......" exit fi
4.3 shell指令碼實戰之一鍵部署lnmp架構、---案例
- 批量部署lnmp架構
[root@web-server01~/script]# vim lnmp.sh #!/bin/bash #install lnmp #by author rivers on 2021-9-28 # nginx 環境準備 Nginx_url=https://nginx.org/download/nginx-1.20.1.tar.gz Nginx_prefix=/usr/local/nginx # mysql 環境準備 Mysql_version=mysql-5.5.20.tar.gz Mysql_dir=mysql-5.5.20 Mysql_url=https://downloads.mysql.com/archives/get/p/23/file/mysql-5.5.20.tar.gz Mysql_prefix=/usr/local/mysql/ # php 環境準備 Php_version=php-7.2.10.tar.gz Php_prefix=/usr/local/php-7.2.10/ function nginx_install(){ if [[ "$1" -eq "1" ]];then if [ $? -eq 0 ];then make && make install fi fi } function mysql_install(){ if [[ "$1" -eq "2" ]];then -DMYSQL_UNIX_ADDR=/tmp/mysql.sock / -DMYSQL_DATADIR=/data/mysql / -DSYSCONFDIR=/etc / -DMYSQL_USER=mysql / -DMYSQL_TCP_PORT=3306 / -DWITH_XTRADB_STORAGE_ENGINE=1 / -DWITH_INNOBASE_STORAGE_ENGINE=1 / -DWITH_PARTITION_STORAGE_ENGINE=1 / -DWITH_BLACKHOLE_STORAGE_ENGINE=1 / -DWITH_MYISAM_STORAGE_ENGINE=1 / -DWITH_READLINE=1 / -DENABLED_LOCAL_INFILE=1 / -DWITH_EXTRA_CHARSETS=1 / -DDEFAULT_CHARSET=utf8 / -DDEFAULT_COLLATION=utf8_general_ci / -DEXTRA_CHARSETS=all / echo -e "/033[32mThe $Mysql_dir Server Install Success !/033[0m" else echo -e "/033[32mThe $Mysql_dir Make or Make install ERROR,Please Check......" exit 0 fi /bin/cp support-files/my-small.cnf /etc/my.cnf /bin/cp support-files/mysql.server /etc/init.d/mysqld chmod x /etc/init.d/mysqld chkconfig --add mysqld chkconfig mysqld on fi } function php_install(){ if [[ "$1" -eq "3" ]];then if [ $? -eq 0 ];then make ZEND_EXTRA_LIBS='-liconv' && make install if [[ "$1" -eq "3" ]];then wget $Php_url && tar xf $Php_version && cd $Php_dir && yum install bxml2* bzip2* libcurl* libjpeg* libpng* freetype* gmp* libm crypt* readline* libxslt* -y && ./configure --prefix=$Php_prefix --disable-fileinfo --enable-fpm --with-config-file-path=/etc --wi -config-file-scan-dir=/etc/php.d --with-openssl --with-zlib --with-curl --enable-ftp --with-gd --with-xmlrpc --with-jpeg-dir --w ith-png-dir --with-freetype-dir --enable-gd-native-ttf --enable-mbstring --with-mcrypt=/usr/local/libmcrypt --enable-zip --enable- mysqlnd --with-mysqli=mysqlnd --with-pdo-mysql=mysqlnd --with-mysql-sock=/var/lib/mysql/mysql.sock --without-pear --enable-bcmath if [ $? -eq 0 ];then make ZEND_EXTRA_LIBS='-liconv' && make install echo -e "/n/033[32m-----------------------------------------------/033[0m" echo -e "/033[32mThe $Php_version Server Install Success !/033[0m" else echo -e "/033[32mThe $Php_version Make or Make install ERROR,Please Check......" exit 0 fi fi } PS3="Please enter you select install menu:" select i in nginx mysql php quit do "lnmp.sh" 113L, 3516C written [root@web-server01~/script]# vim lnmp.sh chkconfig --add mysqld chkconfig mysqld on fi } function php_install(){ if [[ "$1" -eq "3" ]];then if [ $? -eq 0 ];then make ZEND_EXTRA_LIBS='-liconv' && make install echo -e "/n/033[32m-----------------------------------------------/033[0m" echo -e "/033[32mThe $Php_version Server Install Success !/033[0m" else echo -e "/033[32mThe $Php_version Make or Make install ERROR,Please Check......" exit 0 fi fi } PS3="Please enter you select install menu:" select i in nginx mysql php quit do case $i in nginx) nginx_install 1 ;; mysql) mysql_install 2 ;; php) php_install 3 ;; quit) exit esac done
總結
shell是我們工作中很重要的一部分,在工作中充當著重要的角色。企業生產環境中,伺服器規模成百上千,如果依靠人工去維護和管理是非常吃力的,基於Shell程式設計指令碼管理和維護伺服器變得簡單、從容,而且對企業自動化運維之路的建設起到極大的推動作用。
所以,不管是你看到哪裡的書籍、資料、還是視訊,每個老師講的側重點都不一樣,但是基礎是一樣的,要想學好shell,需要反覆動手寫,思考,理解shell的精髓,才能掌握shell ,寫好shell。
來源:
https://developer.aliyun.com/article/946241