Bash腳本編程之變數與多命令執行

来源:https://www.cnblogs.com/alongdidi/archive/2019/12/09/bash_variable_and_multi-cmd-run.html

變數基礎知識 程式由指令加數據所組成,而變數可以理解為數據來源的一種。 變數名可以理解為指向了某個記憶體空間的地址,對於變數的賦值可理解為向記憶體空間寫入數據,對於變數的引用可理解為從記憶體空間讀取數據。 變數有類型的概念(例如字元串、數字等),不同的類型決定了數據的存儲格式、可表示的數據範圍以及可參與的 ...


變數基礎知識

程式由指令加數據所組成,而變數可以理解為數據來源的一種。

變數名可以理解為指向了某個記憶體空間的地址,對於變數的賦值可理解為向記憶體空間寫入數據,對於變數的引用可理解為從記憶體空間讀取數據。

變數有類型的概念(例如字元串、數字等),不同的類型決定了數據的存儲格式、可表示的數據範圍以及可參與的運算等。

在編程語言中,可以根據變數的強弱來劃分。例如C語言屬於強類型變數的語言,該類變數在類型不同的情況下無法直接進行運算。bash腳本則屬於弱類型變數的語言,不同類型的變數可以直接運算,預設將所有的變數統一視為字元型,必須在藉助第三方工具的情況下才可以進行浮點數的運算,變數無需事先聲明即可使用(相當於把聲明、賦值和定義數據類型的操作同時實現)。

變數的聲明就是事先占用好這個記憶體空間。變數的強弱之分與是否需要事先聲明無關,例如python是強變數類型的編程語言但是其變數不需要事先聲明。

變數名可由字母、數字和下劃線組成,並且不允許數字打頭(幾乎所有的編程語言均如此)。變數名要儘量做到見名知意,並且要有一定的規則,根據詞語來劃分(駝峰法、下劃線法等),例如:

my_favorite_sport="basketball"
myFavoriteSport="basketball"

不要使用bash保留字來給變數命名,例如if、else、then等。

變數根據作用域可以劃分為三種:

  • 本地變數:僅當前shell有效(即當前bash進程)。
  • 環境變數:當前shell及其子shell(即當前bash進程即其子bash進程)。
  • 局部變數:在某部分代碼片段中有效(例如函數)。

除了上述三種,還有位置參數變數和特殊變數:

  • 位置參數變數:用於傳參給shell腳本或者函數的變數,例如${1}、${2}等。
  • 特殊變數:在shell中具有特殊含義的變數,例如$?、$-等。

本地變數

本地變數的賦值與引用:

賦值:name=value
引用:${name}或者$name

在我們使用變數的時候,會遇到引號的情況,在bash中,引用(quoting)的作用是抑制bash對某些字元的特殊作用(例如美元符號$,就會做一些變數展開或者特殊符號的處理)。而引號則是引用的其中兩種方式(單引號和雙引號)。

單引號會使得其中的所有字元只有其字面意義,而雙引號只會抑制大部分的特殊字元含義,但是不包括$等其他一些字元。

[[email protected] ~]# name=zwl
[[email protected]-server ~]# echo '${name}'
${name}
[[email protected]-server ~]# echo "${name}"
zwl

在引用變數的時候,也建議使用${name},因為這樣子才可以支持bash的參數展開功能。

[[email protected] ~]# echo ${name}
alongdidi
[[email protected]-server ~]# echo ${name:2:3}
ong
[[email protected]-server ~]# echo $name:2:3
alongdidi:2:3

我們可以通過set(bash內置命令)來查看已經設置的變數名稱和值。不過set所顯示出來的變數非常多,還包含了環境變數和函數,因此一般需要通過管道傳輸給less分頁或者grep過濾。

~]# set | grep "^name"
name=alongdidi

set還可以用於設置shell的屬性(即工作特性)和位置參數,這裡不展開。

當某個變數我們不要的時候,可以使用unset來將其取消掉。

[[email protected] ~]# unset name
[[email protected]-server ~]# echo $name

[[email protected]-server ~]# set | grep "^name"
[[email protected]-server ~]#

unset命令可以取消變數,也可以取消函數。

unset [-f] [-v] [name ...]

-f:指明取消的name是函數。

-v:指明取消的name是變數。

這兩個選項都可以省略,unset會首先嘗試取消一個變數,若失敗則再次嘗試取消一個函數。某些變數無法被unset,例如只讀變數。

可以通過shell內置命令readonly來將shell變數或者函數設置為只讀的屬性。

readonly [-aAf] [name[=value] ...]
readonly -p

第一種語法,是設置一個變數為只讀變數並且可以選擇是否賦值,如果此時不賦值,那麼設置為只讀後就無法再賦值或者改值了。

-a:指明name為索引數組(indexed array)。

-A:指明name為關聯數組(associative array)。

-f:指明name為函數(function)。

[[email protected] ~]# readonly age=28
[[email protected]-server ~]# age=30
-bash: age: readonly variable
[[email protected]-server ~]# unset age
-bash: unset: age: cannot unset: readonly variable

-p:單獨使用,顯示出所有的只讀變數和函數。

[[email protected] ~]# readonly -p
declare -r BASHOPTS="checkwinsize:cmdhist:expand_aliases:extglob:extquote:force_fignore:histappend:interactive_comments:login_shell:progcomp:promptvars:sourcepath"
declare -ir BASHPID
declare -r BASH_COMPLETION_COMPAT_DIR="/etc/bash_completion.d"
declare -ar BASH_VERSINFO='([0]="4" [1]="2" [2]="46" [3]="2" [4]="release" [5]="x86_64-redhat-linux-gnu")'
declare -ir EUID="0"
declare -ir PPID="3530"
declare -r SHELLOPTS="braceexpand:emacs:hashall:histexpand:history:interactive-comments:monitor"
declare -ir UID="0"
declare -r age="28"

註意:declare -r就表示定義一個只讀變數,這也是個shell內置命令,後面會提到。

我們剛說了,本地變數的作用域只在當前的shell中,那麼在子shell和其他shell中應該是不存在的,我們來證明一下。

[[email protected] ~]# pstree -ph 1148
sshd(1148)┬─sshd(1851)───bash(1857)
           ├─sshd(3530)───bash(3536)───pstree(4859)
           └─sshd(4125)───bash(4131)
[[email protected]-server ~]# name=alongdidi
[[email protected]-server ~]# echo $name
alongdidi
[[email protected]-server ~]# bash
[[email protected]-server ~]# pstree -ph 1148
sshd(1148)─┬─sshd(1851)───bash(1857)
           ├─sshd(3530)───bash(3536)───bash(4861)───pstree(4894)
           └─sshd(4125)───bash(4131)
[[email protected]-server ~]# echo $name

[[email protected]-server ~]#

我們原本在PID為3536的shell(bash進程)中,在當前shell設置了變數name並且可獲取其值,隨後使用bash命令進入了子shell(PID為4861的bash進程)中再獲取這個變數,就獲取不到了。

如果我們使用exit命令退回剛纔的父shell,就又可以獲取到該值。

[[email protected] ~]# exit
exit
[[email protected]-server ~]# echo $name
alongdidi

我們切換使用Xshell再創建了一個新的session,在那個session下嘗試獲取該變數的值,也是沒有的。證明成功。

[[email protected] ~]# pstree -ph 1148
sshd(1148)─┬─sshd(1851)───bash(1857)
           ├─sshd(3530)───bash(3536)
           └─sshd(4125)───bash(4131)───pstree(4860)
[[email protected]-server ~]# echo $name

[[email protected]-server ~]#

本地變數的生命周期,從創建的時候開始,直到所在的bash進程結束(比如使用exit)或者變數被unset。

環境變數

環境變數的作用域是當前shell及其子shell。因此也可以理解為將變數輸出(export)到了子shell中。無論子shell的層數有幾層,只要變數具備環境變數的屬性,那麼它就都會有值。

可以通過export或者declare將變數設置為環境變數。

[[email protected] ~]# export name
[[email protected]-server ~]# bash
[[email protected]-server ~]# echo $name
alongdidi
[[email protected]-server ~]# bash
[[email protected]-server ~]# echo $name
alongdidi
[[email protected]-server ~]# bash 
[[email protected]-server ~]# echo $name
alongdidi

export的語法如下。

export [-fn] [name[=value] ...]
export -p

-f:聲明name是一個函數。

-n:去除環境變數的屬性。

-p:使用該選項或者僅使用export命令的話,可以查看當前所有的環境變數。declare -x有同效。外部命令env和printenv也可以實現。

export是專門用於設置與環境變數屬性相關的命令。declare是設置所有與變數相關的屬性(只讀、環境變數、整型、索引數組、關聯數組)等等。

 

多命令執行

;:分號,命令按順序執行,執行完第一個再執行第二個,後一個命令的執行與否與前一個命令的執行成功與否無關。

~]# COMMAND1; COMMAND2

&&:邏輯與,只有當COMMAND1執行成功的時候,才執行COMMAND2。類似於邏輯與運算,a && b,當a為1的時候(執行成功)還需要看b的值(即還需要執行b)才能決定整個表達式的值,如果a為0了,那麼整個表達式必然為0,就不需要看b的值了(b就可以不執行),這也叫短路法則。

~]# COMMAND1 && COMMAND2

||:邏輯或,只有當COMMAND1執行失敗的時候,才執行COMMAND2。

~]# COMMAND1 || COMMAND2

 

 


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

更多相關文章
  • abp依賴註入的原理剖析 請先移步參考 "[Abp vNext 源碼分析] 3. 依賴註入與攔截器" 本文此篇文章的補充和完善。 abp的依賴註入最後是通過IConventionalRegister介面的AddType方法實現的。先看下該介面: 該介面定義了三個方法,支持傳入程式集、類型數組、具體類 ...
  • 使用abp模板進行項目開發: 1:準備工作:我使用v2.x版本,所以會依賴.net core 2.2 查看並更新本地環境: 更新.net core版本:cmd 運行 dotnet --version查看本機版本(我的是core 2.1) 通過下載安裝sdk更新core版本:https://dotne ...
  • 下麵是針對 nfs 所有的版本,我們可以通過不同的RFC 進行詳細看其RFC的細節來進行對比: 下麵是備忘一些NFS RFC 的鏈接: https://datatracker.ietf.org/doc/search?name=nfs&sort=&rfcs=on&activedrafts=on nfs ...
  • 關鍵詞:top、perf、sar、ksar、mpstat、uptime、vmstat、pidstat、time、cpustat、munin、htop、glances、atop、nmon、pcp-gui、collectl。 1. top top是最常用的查看系統資源使用情況的工具,包括CPU、記憶體等等 ...
  • * * * * * (下麵的字體對應) 分 時 日 月 周 星號(*):代表所有可能的值,例如month欄位如果是星號,則表示在滿足其它欄位的制約條件後每月都執行該命令操作。 逗號(,):可以用逗號隔開的值指定一個列表範圍,例如,“1,2,5,7,8,9” 中杠(-):可以用整數之間的中杠表示一個整 ...
  • 準備環境 軟體安裝 服務端基本配置文件 創建一個新的 PKI 和 CA 生成服務端證書 簽約服務端證書 創建 Diffie Hellman 客戶端基本配置 生成客戶端證書 最後簽約客戶端證書 證書整理 服務端 客戶端 server.conf 密碼驗證腳本 !/bin/sh PASSFILE="/et ...
  • 準備環境 環境測試 軟體安裝 xl2tpd.conf options.xl2tpd ipsec.conf l2tp ipsec.conf chap secrets 也就是我們一會在windows上登陸時用到的帳號和密碼 default secrets sysctl.conf 檢查配置 防火牆 開啟服 ...
  • 準備環境 軟體安裝 pptpd.conf options.pptpd chap secrets 也就是我們一會在windows上登陸時用到的帳號和密碼 sysctl.conf 啟動服務並查看埠 防火牆 主機連接 ...
一周排行
  • " 返回《C 併發編程》" "1. 概念介紹" "2. 非同步編程" "2.1. async運行過程" "2.2. async運行中同步上下文簡介" "2.3. 創建Task實例" "2.4. 捕獲非同步異常類型" "3. 並行編程" "3.1. Parallel" "3.2. 異常處理" "3.3. ...
  • 我們先看看兩個特效,感受一下,有沒有學習的動力? 核心API:Texture2D.SetPixel(int x, int y, Color color),Texture2D.Apply() 實現原理:對象池 思路: 第一幀繪製前:遍歷瓦片上所有活著的粒子對象並且進行數據操作(或運動,死亡),發生運動 ...
  • 原來的導出方式比較適用於比較簡單的導出,每一條數據在一行,數據列雖然自定義程度比較高,如果要一條數據對應多行就做不到了,於是就想支持根據模板導出,在 1.8.0 版本中引入了根據模板導出的功能 ...
  • 創建一個bat腳本, 裡面寫上: reg delete HKEY_CURRENT_USER\Software\JetBrains\dotMemory /freg delete HKEY_CURRENT_USER\Software\Microsoft\Windows\CurrentVersion\Ex ...
  • Linux下有vsyscall來優化一些例如time(NULL), gettimeofday這種調用的消耗; 但是Windows下, 沒有類似的東西, 但是思路還是有的 1. 程式啟動的時候, 獲取一下準確的時間戳 2. 然後每次需要獲取時間的時候, 獲取一下流逝的時間, 可以通過獲取CPU的tic ...
  • 當用戶按下鍵盤上的一個鍵時,就會發生一系列事件。下表根據他們的發生順序列出了這些事件: 表 所有元素的鍵盤事件(按順序) 鍵盤處理永遠不會像上面看到的這麼簡單。一些控制項可能會掛起這些事件中的某些事件,從而可執行自己更特殊的鍵盤處理。最明顯的例子是TextBox控制項,它掛起了TextInput事件。對 ...
  • static void LocalMethod() { Cube(100); void Cube(int x) => Console.WriteLine($"The cube of {x} is {x * x * x}"); } static void GoToDemo() { int i = 1; ...
  • 滑鼠事件執行幾個關聯的任務。當滑鼠移到某個元素上時,可通過最基本的滑鼠事件進行響應。這些事件是MouseEnter(當滑鼠指針移到元素上時引發該事件)和MouseLeave(當滑鼠指針離開元素時引發該事件)。這兩個事件都是直接事件,這意味著他們不使用冒泡和隧道過程,而是源自一個元素並且只被該元素引發 ...
  • 反射這個詞聽起來就很牛逼是吧? 嗯的確,反射是比較高級的特性,只有語言基礎很扎實的Dev們才應該使用它。 搞點反射,可以提高程式的靈活性、可擴展性、耦合度。 反射這東西,是為了動態地運行時載入,相比於靜態代碼。編譯的時候就是板上釘釘了。 就是說,如果你的程式需要在運行時搞一些晚綁定,動態載入或檢查對 ...
  • 眾所周知,微服務架構是由一眾微服務組成,項目中調用其他微服務介面更是常見的操作。為了便於調用外部介面,我們的常用思路一般都是封裝一個外部介面的客戶端,使用時候直接調用相應的方法。webservice或WCF的做法就是引用服務,自動生成客戶端。在webapi2.0里,我們都會手動封裝一個靜態類。那麼在 ...
x