13種Shell邏輯與算術,能寫出5種算你贏!

来源:https://www.cnblogs.com/jiagooushi/archive/2023/01/10/17040141.html
-Advertisement-
Play Games

相較於最初的 Bourne shell,現代 bash 版本的最大改進之一體現在算術方面。早期的 shell 版本沒有內建的算術功能,哪怕是給變數加1,也得調用單獨的程式來完成。 1、算術方法一: $(( )) 只要都是整數運算,就可以在 $(( )) 的算術表達式內使用所有的標準運算符。還有一個額 ...


相較於最初的 Bourne shell,現代 bash 版本的最大改進之一體現在算術方面。早期的 shell 版本沒有內建的算術功能,哪怕是給變數加1,也得調用單獨的程式來完成。

1、算術方法一: $(( ))

只要都是整數運算,就可以在 $(( )) 的算術表達式內使用所有的標準運算符。還有一個額外的運算符:可以用** 進行冪運算,如下:

COUNT=$((COUNT + 5 + MAX * 2))

或者:

MAX=$((2**8))

$(( )) 表達式內不需要使用空格,不過在運算符和操作數兩邊加上空格也無妨(但 ** 必須寫在一起)。但是 = 兩邊絕不能出現空格,這和 bash 變數賦值的規則一樣。如果你按以下方式寫:

COUNT = $((COUNT+5))   # 註意 = 號兩邊多了空格,可不像你想的那樣!

那麼,bash 會嘗試運行一個名為 COUNT 的程式,其第一個參數為 =,第二個參數為 $COUNT 與 5 之和。記住,別在賦值號兩邊加空格!

另一個怪異之處是,通常出現在 shell 變數前表示取值的 $ 符號(如 $COUNT 或 $MAX)在雙括弧內部是不需要的。例如,我們可以寫:

$((COUNT + 5 + MAX * 2))

shell 變數前並沒有 $ 符號,實際上,外部的 $ 應用於整個表達式。但如果用到了位置參數(如 $2),那麼 $ 還是少不了的,因為只有這樣才能區分位置參數與數字常量(如 2)。以下是一個示例。

COUNT=$((COUNT + $2 + OFFSET))

也可以用逗號運算符形成級聯賦值,如下圖:

echo $(( X+=5 , Y*=3 ))

該表達式執行兩次賦值操作,然後由 echo 顯示出第二個子表達式的結果(因為逗號運算符返回其第二個操作數的值)。

2、算術方法二:let

除去使用$(())可進行算術運算外,還可以使用let語句,如下:

let COUNT=COUNT+5

同$(())一樣,在使用變數時不需要使用$符號。但是,當我們需要使用let進行COUNT=$((COUNT + 5 + MAX * 2))格式的運算時,需要使用到引號‘’,如下:

let COUNT+='5+MAX*2'

let 語句和 $(( )) 語法的另一處重要區別在於兩者處理空白字元(空格字元)的方式不同。對 let 語句來說,要麼添加引號,要麼賦值運算符(=)和其他運算符兩邊不能出現空格。必須將運算符和操作數放在一起形成一個單詞。以下兩種寫法都沒問題。

let i=2+2
let "i = 2 + 2"

$(( )) 語法就寬鬆多了,它允許各種空白字元出現在雙括弧內。這種寫法不易出錯,代碼的可讀性也要好得多,是我們執行 bash 整數運算時的首選方式。

3、bash中的賦值運算符

file

4、條件分支if

條件判斷,邏輯分支是任何一個語言都會遇到的問題,bash中同其他語言類似,都是使用if進行條件判斷,如下:

if [ $# -lt 3 ]
then
 printf "%b" "Error. Not enough arguments.\n"
 printf "%b" "usage: myscript file1 op file2\n"
 exit 1
fi

或者:

if (( $# < 3 ))
then
 printf "%b" "Error. Not enough arguments.\n"
 printf "%b" "usage: myscript file1 op file2\n"
 exit 1
fi

以下是一個帶有 elif(bash 中的 else-if)和 else 子句的完整if 語句。如下:

if (( $# < 3 ))
then
 printf "%b" "Error. Not enough arguments.\n"
 printf "%b" "usage: myscript file1 op file2\n"
 exit 1
elif (( $# > 3 ))
then
 printf "%b" "Error. Too many arguments.\n"
 printf "%b" "usage: myscript file1 op file2\n"
 exit 2
else
 printf "%b" "Argument count correct. Proceeding...\n"
fi

關於if,我們有兩個問題需要明白,分別是:

  • if 語句的基本結構
  • if 表達式的不同語法(括弧或方括弧,運算符或選項)

5、if的基本結構

按照 bash 手冊頁中的描述,if 語句的一般形式如下所示。

if list; then list; [ elif list; then list; ] ... [ else list; ]
fi

[ 和 ] 用於劃分語句中的可選部分(例如,有些 if 語句中就沒有else 子句)。我們先來看看不帶任何可選部分的 if 語句。

最簡單的 if 語句形式如下所示。

if list; then list; fi

在 bash 中,和換行符一樣,分號的作用也是結束某個語句。我們可以用分號將解決方案部分中的示例塞進更少的行中,但使用換行符的可讀性更好。then list 的存在看起來是有意義的,其中的語句在 if 條件為真的情況下執行(我們也可以從其他編程語言中猜測出來)。但是,if list 算是怎麼回事?難道不應該是 if expression 嗎?

沒錯,但這是 shell,一個命令處理器。它的主要任務就是執行命令。因此,if 後面的 list 就是放置命令列表的地方。你可能會問,決定分支走向(then 子句或 else 子句)的是什麼呢?答案是list 中最後一個命令的返回值

我們通過一個有點奇怪的示例來說明這一點。如下:

$ cat trythis.sh // 查看腳本內容,如下所示

if ls; pwd; cd $1; 

then

 echo success

else

 echo failed

fi

// 執行腳本,傳遞一個參數
$ bash ./trythis.sh /tmp

在這個奇怪的腳本中,shell 會在選擇分支前執行 3 個命令(ls、pwd、cd),其中 cd 命令的參數是調用該腳本時所提供的第一個命令行參數。如果沒有提供參數,那就只執行 cd,返回到主目錄中。結果會是怎樣?你可以自己試試。最終是顯示“success”還是“failed”,取決於 cd 命令是否執行成功。在示例中,cd 是 if語句命令列表中的最後一個命令。如果 cd 執行失敗,就轉到 else子句;但如果執行成功,則選擇 then 子句。

6、if中的 [] 和 (())

我們一起來看下麵的例子:

if test $# -lt 3
then
 echo try again.
fi

前面講到if後面是list是命令列表,雖然此處不是命令列表,但有沒有從中看出起碼類似於單個 shell 命令(內建命令 test 接受參數並比較參數值)的東西?

在本章開頭,我們給出的第一個示例中開頭的 if [ $# -lt 3 ] 看起來很像test 命令。這是因為 [ 其實只是相同命令的不同名稱而已。(出於可讀性和美觀方面的考慮,調用 [ 時還要求將 ] 作為最後一個參數。)因此,對於該語法,if 語句中的表達式其實就是一個只包含單個命令(test 命令)的列表。

在早期的 Unix 中,test 是一個獨立的可執行文件,[ 只是指向該文件的鏈接。現在兩者仍以可執行文件的形式存在,但bash 也將它們實現為內建命令。

那麼 if (( $# < 3 )) 又是什麼意思?

雙括弧是複合命令的一種。因為它會對其中的算術表達式求值,所以能在 if 語句中派上用場。這是一處比較新的 bash 改進 ,專門用於有 if 語句的場合。

可用於 if 語句的這兩種語法之間的重要區別在於測試的表達方式及其能夠測試的對象種類

雙括弧僅限於算術表達式,方括弧還可以測試文件特性,但後者的算術測試語法遠不如前者方便,尤其是用括弧將表達式劃分成若幹子表達式時。

當我們使用 [ ] 時,一定要註意空格是必須存在的,如下圖:

if [ -d "/opt/" ]

7、測試文件的特性

為了提高腳本的穩健性,你希望在讀取輸入文件前先檢查該文件是否存在;另外,還想在寫入輸出文件前確認其是否具備寫許可權,在用cd 切換目錄前看看到底有沒有這個目錄。這些該如何在 bash 腳本中實現呢?如下所示:

#!/usr/bin/env bash
# 實例文件:checkfile
#
DIRPLACE=/tmp
INFILE=/home/yucca/amazing.data
OUTFILE=/home/yucca/more.results

if [ -d "$DIRPLACE" ] // 判斷是否目錄
then
    cd $DIRPLACE
    if [ -e "$INFILE" ] // 判斷文件是否存在
    then
        if [ -w "$OUTFILE" ] // 判斷文件是否擁有寫許可權
        then
        	doscience < "$INFILE" >> "$OUTFILE"
        else
       		echo "cannot write to $OUTFILE"
        fi
    else
    	echo "cannot read from $INFILE"
    fi
else
	echo "cannot cd into $DIRPLACE"
fi

將各種文件名引用全都放入了引號,以防路徑名中包含空格。在上面的例子,我們使用了測試文件是否是目錄(-d)、文件是否存在(-e)、文件是否有寫許可權(-w),我們也可以測試一些別的文件特性,其中有 3 個特性要用到雙目運算符(接受兩個文件名)。

  • FILE1 -nt FILE2 是否更新(檢查文件的修改時間)。現有文件要比不存在的文件“新”。
  • FILE1 -ot FILE2 是否更舊。同樣,不存在的文件要比現有文件“舊”。
  • FILE1 -ef FILE2 具有相同設備和 inode 編號(即便由不同鏈接所指向,也視為相同的文件)

前面使用的-e、-d、-w都屬於單目運算符,其形式為 option filename,例如,if [ -e myfile ]

8、測試多個特性

前面,我們測試每個特性都是使用單獨一個if語句,那麼我們測試多個特性時,必須嵌套if語句嗎?

使用 -a(邏輯與)和 -o(邏輯或)運算符將多個測試條件組合成一個表達式。例如:

if [ -r $FILE -a -w $FILE ]

該 if 語句會測試指定文件是否可讀並且可寫。

測試時,為啥不加上-e呢?因為所有的文件測試條件都隱含了該文件存在的測試,所以測試文件可讀性時不用測試文件是否存在。如果文件不存在,自然也就不可讀。這些邏輯運算符(-a 表示 AND,-o 表示 OR)可用於所有的測試條件,並不局限於文件測試。

同一個語句中可以出現多個 AND/OR。你可能要用括弧來獲得正確的優先順序,比如 a and (b or c),但一定要記得在括弧前加上反斜杠或將括弧放進引號,以消除其特殊含義。如下:

if [ -r "$FN" -a \( -f "$FN" -o -p "$FN" \) ]

9、測試字元串特性

你希望在使用字元串前先檢查一下它們的值。這些字元串可以是用戶輸入、讀入的文件或傳入腳本的環境變數。如何用 bash 腳本實現呢?

你可以在 if 語句中使用單方括弧形式的 test 命令進行一些簡單的測試,其中包括檢查變數是否包含文本以及兩個變數中的字元串是否相同。如下腳本所示:

# 使用命令行參數
VAR="$1"
#
# if [ "$VAR" ]這種形式通常也管用,但並不是一種好的寫法,加上-n會更清晰

if [ -n "$VAR" ]
then
	echo has text
else
	echo zero length
fi

if [ -z "$VAR" ]
then
	echo zero length
else echo has text
fi

長度為 0 的變數有兩種:設置為空串的變數和不存在的變數。示例中的測試並不區分這兩種情況。它只關心變數中是否有字元存在。

重要的是要將 $VAR 放進引號,否則測試會被一些怪異的用戶輸入干擾。如果 $VAR 的值是 x -a 7 -lt 5 且沒有使用引號,那麼下列語句:

if [ -z $VAR ]

就會變成(在變數擴展之後):

if [ -z x -a 7 -lt 5 ]

10、測試等量關係

你想要檢查兩個 shell 變數是否相等,但是存在兩種測試運算符:-eq 和 =(或 ==)。該用哪個呢?

你需要的比較類型決定了該用哪種運算符。

  • 如果是進行數值比較,可以使用 -eq 運算符。
  • 如果是進行字元串比較,則使用 =(或 ==)運算符。

下麵,我們通過一個簡單的腳本例子來演示,如下:

#
# 老生常談的字元串與數值比較
#

VAR1=" 05 "
VAR2="5"
printf "%s" "do they -eq as equal? "
if [ "$VAR1" -eq "$VAR2" ]
then
	echo YES
else
	echo NO
fi

printf "%s" "do they = as equal? "

if [ "$VAR1" = "$VAR2" ]
then
	echo YES
else
	echo NO
fi

如果,我們運行腳本,則會得到如下結果:

$ ./腳本名
do they -eq as equal? YES
do they = as equal? NO
$

儘管兩個變數的數值相等(5),但從字元角度來看,前導字元 0 和空白字元意味著這兩個字元串並不相同。

= 和 == 都可以使用,但 = 符合 POSIX 標準,可移植性更好。

使用if,我們可以在腳本中進行分支判斷。但是對於系統而言,迴圈同分支一樣是常見需求。所以Shell一樣支持迴圈操作

11、迴圈一段時間

對於算術條件,使用 while 迴圈:

while (( COUNT < MAX )) // 判斷條件是否成立
do // 語法要求,以do開始
 some stuff
 let COUNT++
done // 語法要求,以done結束

對於文件系統相關的條件:

while [ -z "$LOCKFILE" ]
do
 some things
done

第一個 while 語句中的雙括弧界定了算術表達式,這很像 shell 變數賦值中用到的 $(( ))。雙括弧內出現的變數名錶示取值。也就是說,不需要寫成 $VAR,直接在括弧中使用 VAR就行了。

while [ -z"$LOCKFILE" ] 中的方括弧和 if 語句中的一樣,等同於使用 test 命令。

使用(( )) 時,shell 會對其中的表達式求值,如果結果為非0,那麼 (( )) 就返回 0;如果結果為 0,則返回 1。這意味著我們可以像 Java 或 C 程式員那些書寫表達式,但 while 語句沿用的仍舊是 bash 那一套,視 0 為真。實際上,這意味著我們可以編寫一個無限迴圈:

while (( 1 ))
do

 ...dosomething

done

12、迴圈若幹次

如果需要迴圈夠一定次數。可以使用 while 迴圈,在計數時進行測試,不過編程語言中的 for 迴圈正是針對這種情況設計的。那麼,如何在 bash 中實現呢?

使用 for 迴圈語法的一種特例,看起來和 C 語言中的差不多,但使用的是雙括弧。

for (( i=0 ; i < 10 ; i++ )) ; do echo $i ; done

在早期的 shell 版本中,for 迴圈只能按照固定的列表項進行迭代。和文件名之類的打交道時,shell 腳本是面向單詞的,就此而言,這算得上是一個不錯的創新。但如果需要計數,用戶會發現自己可能寫出瞭如下代碼。

for i in 1 2 3 4 5 6 7 8 9 10
do
 echo $i
done

看起來還行,尤其是迴圈次數不多時。可是說實話,換成 500 次迴圈可就不好使了。

bash 2.04 版開始引入一種 for 迴圈的變體,語法與 C 語言類似。其一般形式如下所示。

for (( expr1 ; expr2 ; expr3 )) ; do list ; done

雙括弧表明這是算術表達式,在其中引用變數時,不用加 $(但 $1等位置參數除外),只要是 bash 中出現雙括弧的地方,均是如此。該表達式是整數表達式,可以使用包括逗號(用於在一個表達式中放入多個操作)在內的大量運算符。

for (( i=0, j=0 ; i+j < 10 ; i++, j++ ))
do
 echo $((i*j))
done

for 迴圈先初始化了兩個變數($i 和 $j),然後在第二個更複雜的子表達式中對 $i 和 $j 求和,接著判斷是否小於 10。第三個子表達式再次用逗號運算符累加這兩個變數。

13、在迴圈中使用浮點值

帶有算術表達式的 for 迴圈只能執行整數運算。如果是浮點值,該怎麼辦呢?

如果系統提供了 seq 命令,則可以用它來生成浮點值。

for fp in $(seq 1.0 .01 1.1)
do
 echo $fp; other stuff too
done

seq 命令會生成一系列浮點值,每行一個。該命令的參數依次是起始值、增量、結束值。$() 在子 shell 中執行命令,返回結果中的換行符會被空白字元替換,因此,就 for 迴圈而言,每個值都是字元串。

本文由傳智教育博學谷教研團隊發佈。

如果本文對您有幫助,歡迎關註點贊;如果您有任何建議也可留言評論私信,您的支持是我堅持創作的動力。

轉載請註明出處!


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 儘量不要用== ==的比較過程 以 [] == ![] 為例 左邊 [] 是個數組 右邊 ![] 是個表達式,表示對數組取反,需要對數組進行布爾判定,數組是對象所以是true,取反之後為false,所以就相當於比較 == 兩邊數據類型不一樣會將數據轉變成原始類型 ,原始類型有 null 、 unde ...
  • 路線規劃 簡介 路線規劃常用於出行路線的提前預覽,我們提供4種類型的路線規劃,分別為:駕車、步行、公交和騎行,滿足各種的出行場景。 高德開放平臺 本例是駕車路線規劃功能和位置選擇地圖api:chooseLocation 示例: 1、在頁面的 js 文件中,實例化 AMapWX 對象,請求進行駕車路線 ...
  • 前一段時間做項目,頻繁使用到上傳圖片組件,而且只上傳一個封面,於是想著自定義一個圖片封面上傳組件。先來看一下效果: 第一張圖片是上傳之前,第二張圖片是上傳成功後,第3張圖片是滑鼠放上去之後的效果! 首先整理需求,圖片上傳我們使用照片牆的方式,只能上傳一張圖片,圖片上傳成功後不能繼續上傳,如果想要更換 ...
  • 電銷是什麼?就是坐席拿著電話給客戶打電話嗎?no no no,讓我們一起走進京音平臺之電銷系統。 京音平臺2020年初開始建設,過去的兩年多的時間里,經歷了跌宕起伏,有經驗、有教訓,整體來說平臺經歷了人工、自動化階段,目前處於初步智能化階段,希望可以將過去的一些心路歷程分享給大家,共同交流、共同進... ...
  • 1高階函數 1.1 數學概念 回顧下數學知識: y=f(x) 這是最開始接觸的普通函數 y=g(f(x)) 這個就是我們接觸到的高階函數 在數學和電腦科學中,高階函數至少應當是滿足下麵一個條件的函數: 1)接受一個或者多個函數作為參數 2)輸出一個函數 程式中我們的高階函數也類似 示例計數器的函數 ...
  • Redis 數據結構-雙向鏈表 最是人間留不住,朱顏辭鏡花辭樹。 1、簡介 Redis 之所以快主要得益於它的數據結構、操作記憶體資料庫、單線程和多路 I/O 復用模型,進一步窺探下它常見的五種基本數據的底層數據結構。 Redis 常見數據類型對應的的底層數據結構。 String:簡單動態字元串。 L ...
  • 前言 今天給大家介紹的是Python爬蟲批量下載相親網站圖片數據,在這裡給需要的小伙伴們代碼,並且給出一點小心得。 首先是爬取之前應該儘可能偽裝成瀏覽器而不被識別出來是爬蟲,基本的是加請求頭,但是這樣的純文本數據爬取的人會很多,所以我們需要考慮更換代理IP和隨機更換請求頭的方式來對相親網站圖片數據進 ...
  • 一、MybatisPlusMax簡介 MybatisPlusMax是MybatisPlus的增強包,秉承只拓展不修改的理念,對MybatisPlus做增強。 正如MybatisPlus是對MyBatis的增強,MybatisPlusMax是對MybatisPlus的增強,拓展理念一脈相承。 Myba ...
一周排行
    -Advertisement-
    Play Games
  • 前言 在我們開發過程中基本上不可或缺的用到一些敏感機密數據,比如SQL伺服器的連接串或者是OAuth2的Secret等,這些敏感數據在代碼中是不太安全的,我們不應該在源代碼中存儲密碼和其他的敏感數據,一種推薦的方式是通過Asp.Net Core的機密管理器。 機密管理器 在 ASP.NET Core ...
  • 新改進提供的Taurus Rpc 功能,可以簡化微服務間的調用,同時可以不用再手動輸出模塊名稱,或調用路徑,包括負載均衡,這一切,由框架實現並提供了。新的Taurus Rpc 功能,將使得服務間的調用,更加輕鬆、簡約、高效。 ...
  • 順序棧的介面程式 目錄順序棧的介面程式頭文件創建順序棧入棧出棧利用棧將10進位轉16進位數驗證 頭文件 #include <stdio.h> #include <stdbool.h> #include <stdlib.h> 創建順序棧 // 指的是順序棧中的元素的數據類型,用戶可以根據需要進行修改 ...
  • 前言 整理這個官方翻譯的系列,原因是網上大部分的 tomcat 版本比較舊,此版本為 v11 最新的版本。 開源項目 從零手寫實現 tomcat minicat 別稱【嗅虎】心有猛虎,輕嗅薔薇。 系列文章 web server apache tomcat11-01-官方文檔入門介紹 web serv ...
  • C總結與剖析:關鍵字篇 -- <<C語言深度解剖>> 目錄C總結與剖析:關鍵字篇 -- <<C語言深度解剖>>程式的本質:二進位文件變數1.變數:記憶體上的某個位置開闢的空間2.變數的初始化3.為什麼要有變數4.局部變數與全局變數5.變數的大小由類型決定6.任何一個變數,記憶體賦值都是從低地址開始往高地 ...
  • 如果讓你來做一個有狀態流式應用的故障恢復,你會如何來做呢? 單機和多機會遇到什麼不同的問題? Flink Checkpoint 是做什麼用的?原理是什麼? ...
  • C++ 多級繼承 多級繼承是一種面向對象編程(OOP)特性,允許一個類從多個基類繼承屬性和方法。它使代碼更易於組織和維護,並促進代碼重用。 多級繼承的語法 在 C++ 中,使用 : 符號來指定繼承關係。多級繼承的語法如下: class DerivedClass : public BaseClass1 ...
  • 前言 什麼是SpringCloud? Spring Cloud 是一系列框架的有序集合,它利用 Spring Boot 的開發便利性簡化了分散式系統的開發,比如服務註冊、服務發現、網關、路由、鏈路追蹤等。Spring Cloud 並不是重覆造輪子,而是將市面上開發得比較好的模塊集成進去,進行封裝,從 ...
  • class_template 類模板和函數模板的定義和使用類似,我們已經進行了介紹。有時,有兩個或多個類,其功能是相同的,僅僅是數據類型不同。類模板用於實現類所需數據的類型參數化 template<class NameType, class AgeType> class Person { publi ...
  • 目錄system v IPC簡介共用記憶體需要用到的函數介面shmget函數--獲取對象IDshmat函數--獲得映射空間shmctl函數--釋放資源共用記憶體實現思路註意 system v IPC簡介 消息隊列、共用記憶體和信號量統稱為system v IPC(進程間通信機制),V是羅馬數字5,是UNI ...