Linux Shell腳本

来源:https://www.cnblogs.com/heyongshen/archive/2022/07/10/16464365.html
-Advertisement-
Play Games

基礎概念: - 機器語言:二進位的0和1的序列,稱為機器指令 - 彙編語言:用一些符號來代替機器語言 - 電腦:只能運行二進位指令 - 機器代碼文件:由0和1組成的二進位文件 - 編譯:將編寫好的程式文件通過一個工具(編譯器)變成機器代碼文件,然後在變成可執行文件。 #編譯的特點:效率高 - 解釋 ...


基礎概念:

- 機器語言:二進位的0和1的序列,稱為機器指令

- 彙編語言:用一些符號來代替機器語言

- 電腦:只能運行二進位指令

- 機器代碼文件:由0和1組成的二進位文件

- 編譯:將編寫好的程式文件通過一個工具(編譯器)變成機器代碼文件,然後在變成可執行文件。
#編譯的特點:效率高

- 解釋:將編寫好的程式文件中的代碼一行一行按順序通過一個工具(解釋器)解釋為二進位指令。
#解釋的特點:需要把源代碼交給使用者

- 編程語言分類:低級語言和高級語言
#高級語言需要經過解釋或編譯才能轉化為電腦能識別的語言

- 編程風格:面向對象和麵向過程

shell腳本;

shell:是面向過程、解釋型語言(需要指定解釋器)

shell腳本的格式要求:

首行遵循shebang(#!)機制

執行shell腳本:

  • 通過bash命令來執行(支持標準輸入和輸出)

  • 通過添加執行許可權來執行

shell中的變數:

  • 自定義變數:

  • 內置變數:比如PATH、HOSTNAME、PS1等

shell變數按生效範圍等標準劃分:

  • 普通變數:生效範圍只是當前的shell

  • 環境變數:當前shell以及shell的子進程

  • 本地變數:函數中

shell中變數的類型:

  • 字元型

  • 整型、浮點型

靜態語言和動態語言

  • 靜態語言使用變數前需要先定義變數的類型

  • 動態語言不用定義變數的類型

強類型和弱類型語言:

  • 強類型語言不同類型的數據運算需要經過強制轉換

  • 弱類型語言:會做隱式轉換

shell屬於弱類型、動態語言。所以shell變數申明的時候不需要指定變數類型,且不同類型格式進行計算的時候不需要進行強制轉換。

shell變數的定義規則:

  • 區分大小寫

  • 內置變數和保留字不能用

  • 數字字母下劃線(數字不能用作開頭)(定義主機名的時候不能使用‘-’)

shell定義變數:

變數名='值'(等號兩邊不能有空格) #如果值是字元串最好用引號括起來
#變數臨時生效,推出shell後,變數自動刪除。

shell查看和取消變數

  • 查看所有變數: set命令

  • 取消指定的變數: unset 變數名

shell引用變數

通過$name或${name}

單引號、雙引號、反引號的區別:

  • 雙引號:屬於弱引用,裡面的變數會替換成對應的值

  • 單引號:強引用,裡面是什麼,顯示的就是什麼

  • 反引號:命令替換,將反引號內的Linux命令先執行

shell引用變數:

$變數名或${變數名}

shell中的位置變數

是bash內置的變數,通過命令行將參數傳遞給shell腳本的對應參數

$n       #表示指定位置的參數,比如$1表示的就是shell腳本第一個位置的參數值

$0       #表示腳本文件的名稱

$*       #表示shell腳本所有的參數

$#       #表示shell腳本參數的個數

shell的退出狀態碼: $?

進程執行完畢以後,會使用$?來保存這個進程的狀態碼,狀態碼的取值範圍是:0--255

0      #表示執行成功:

1-255  #表示執行失敗(因為執行失敗有很多可能性,比如語法錯誤、命令沒找到等,所以狀態碼不一樣。)

shell自定義退出的狀態碼;

定義方法: exit [num]

#腳本中一旦遇到exit命令,腳本會立即終止;終止退出狀態取決於exit命令後面的數字

#如果exit後面無數字,終止退出狀態取決於exit命令前面命令執行結果

#如果沒有exit命令,整個腳本的退出狀態碼取決於腳本中執行的最後一條命令的狀態碼

shell腳本的調試和安全

shell腳本調試

#語法錯誤:會導致後續命令不執行
使用bash -n 只檢測腳本中的語法錯誤,但無法檢查出命令錯誤(不真正執行腳本)

#命令錯誤:後續命令依然會執行
使用bash -x 來跟蹤命令的執行過程。

#邏輯錯誤:輸出結果不對
使用bash -x 檢查

shell腳本安全防範

set -e:如果其中一個命令錯誤,就中斷執行
#有時候會產生誤殺,比如執行一條命令返回的狀態碼是非0

set -u: 當一個變數沒有定義的時候就退出。避免變數不存在出現的問題

範例;編寫第一個shell腳本

# 1.編寫shell腳本文件
#!/bin/bash   ##首行shebang機制,指定使用的解釋器
echo 'hell shell' #編寫需要做的事
#註:shell腳本中使用#號進行語句的註釋

# 2.給腳本文件添加執行許可權或者使用bash命令執行腳本
#添加執行許可權的方法:
chmod +x shell_name
path/shell_name #路徑/文件名

#bash命令的方法
bash shell_name

shell腳本實現算數運算和邏輯運算

shell腳本的算數運算

shell腳本實現算數運算的方法;

  • let工具

  • bc工具(支持小數運算)

  • expr工具

  • 運算語法:

shell的運算語法:

  • $[算數表達式]

  • $((算數表達式))

shell算術運算工具的使用:

let工具:

let用於執行一個或多個表達式,變數計算中**不需要加 $ **來表示變數
註意:這種方法需要藉助一個變數,將計算後的值賦值給這個變數後進行輸出

let a=1+1

bc工具

支持標準輸入和輸出

echo 1+1 | bc

expr工具

expr 1 + 1 
#數值與運算符號之間需要用空格隔開
#使用expr命令進行乘法運算時,需要將”乘號”轉義

運算語法


#$[算數表達式]:
echo $[10+1]

#$((算數表達式))
echo $((1+1))
將變數申明為整數進行計算
#直接將變數聲明為”整數”,然後即可直接進行整數運算
declare -i a
a=1+1
echo $a

shell腳本的邏輯運算

  • 與:&

  • 或:|

  • 非:!

  • 異或:^(脫字元)

短路運算

  • 短路與 &&:前面為真才會執行後面的

  • 短路或 ||:前面為真後面的就不執行了

邏輯運算的結果

邏輯運算的結果是一個布爾值。true或false

  • true:1(二進位),真

  • false:0(二進位),假

#註意:$?(退出狀態碼) 為0 ,表示為真,非0則表示為假

shell的條件測試語句

test命令:

用於檢查某個條件是否成立,它可以進行數值、字元和文件三個方面的測試

test命令的格式

  • 格式一:test 表達式

  • 格式二: [ 表達式 ] (中括弧的兩邊要有空格作為間隔)

  • 格式三: [[ 表達式 ]] (且支持擴展正則表達式和通配符)

#註意:test和中括弧等價
[root@centos8 ~]#help [
[: [ arg... ]
   Evaluate conditional expression.   
   This is a synonym for the "test" builtin, but the last argument must
   be a literal `]', to match the opening `['.

使用test判斷文件的許可權

  • -r:是否存在讀許可權

  • -x:是否存在執行許可權

  • -w:是否存在寫許可權
    範例:判斷一個文件是否存在讀許可權

test -r aa; echo $? 或 [ -r aa ];echo $?
#若存在,退出狀態碼為0

文件類型判斷

  • -f:是否為文本文件

  • -d:是否為目錄文件

文件狀態判斷

  • -e:文件是否存在

  • -a:文件是否存在

字元串判斷:

  • -z STRING: 字元串是否為空,沒定義或空為真,不空為假

  • = != > <:連個字元串是否相同、不同等。(這些符號兩邊要有空格)

數字之間比較

  • -eq: equal,相等

  • -ne: not-equal,不等

  • -gt: greater-than,大於

  • -lt:less-than,小於

[[ expression ]] 用法

使用通配符的時候: == 和 !=

會把右側的認為是通配符

使用正則表達式的時候: =~

會把右側的認為是正則表達式(擴展正則表達式)

組合條件測試

將兩個條件通過並且、或者連接起來

  • 並且: -a

  • 或者: -o

範例:

#如果當前用戶非root且該文件不存在就創建這個文件
[ $(whoami) != 'root' -a ! -e /data/dir ] && sudo mkdir /data/dir

read命令

接受鍵盤的輸入,然後把他分配給一個指定的變數。

#若沒有指定變數來接受read得到的數據,則預設賦值給內置的變數REPLY
#判斷用戶輸入的是否為YES
#!/bin/bash
read -p 'continute?yes or no:'  ANSWER
[[ $ANSWER  =~  ^([yY][eE][sS])$ ]]  && echo "let's continue" || echo "stop"
#檢查主機的網路狀態

#!/bin/bash
read -p "please input IP:" IP
[[ "$IP" =~  ^[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}\.[0-9]{1,3}$  ]] && echo "$IP is availabe"
ping -c 4 $IP > /dev/null && echo "$IP is Up " || echo "$IP is unreachable"
選項;

-p:指定提示的信息

()和{}

作用:將多個命令組合在一起,批量執行
格式:( cmd;cmd2....)或{ cmd,cmd2...} --括弧左邊要有空格

#()和{}批量執行命令的區別:使用小括弧(),裡面執行的命令會開啟子進程,大括弧不會

image
image

範例:

#判斷網路的狀態
IP="x.x.x.x" #賦值的時候等號兩邊不能有空格   -c:指定ping的次數
ping -c 4 $IP &> /dev/null  && echo "$IP is up"  || echo "$IP is unreachable";
echo "scripts is finished"
exit
#範例:磁碟空間的判斷,判斷某塊磁碟的使用空間是否達到臨界值

WARING=80 #設置一個臨界值
SPACE_USE=`df|grep '^/dev/sd'|tr -s ' ' %|cut -d% -f5|sort -nr|head -1`
[ "$SPACE_USED" -ge $WARNING ] && echo "disk used is $SPACE_USED,will be full"


#tr命令
#作用: 字元轉換和刪除
#選項:
-d:delete
-s:squeeze-repeats
#範例:
轉換:tr '123' 'abc' #把132轉換成abc -- 一 一對應 1--a 2--b
刪除: tr -d '需要刪除的內容'
將多個連續的字元壓縮或轉換成一個指定字元:tr -s '' #把多個連續的空格壓縮成一個空格(-s:把相同連續的字元合成一個)

cut命令:
作用:提取列
選項:
#-d -- 指定分隔符
#-f -- 指定提取取那些列

head命令:
作用:顯示文件的前幾行
選項:
-n:指定獲取前x行

shell腳本的流程式控制制語句

  • 條件選擇

  • 條件判斷

  • 迴圈語句

shell的條件選擇語句:if

  • 單分支

  • 雙分支

  • 多分支

單分支格式

if 判斷條件;then
   執行的操作
fi

雙分支格式

if 判斷條件;then
   執行的操作
else
   執行的操作
fi

多分支

if 判斷條件;then
   執行的操作
elif
   執行的操作
   ......
else
   執行的操作
fi

範例

#根據命令的退出狀態來執行命令
#/bin/bash
read IP
if ping -c 2 $IP &> /dev/null;then
 	echo "$IP is up"
else
	echo "$IP is unreachable"
fi

條件判斷 case 語句

case 變數引用 in 
PAT1)
	執行相關操作
	;;
PAT2)
	執行相關操作
	;;
*)
	執行相關操作
	;;
esac

#case語句支持通配符
* 任意長度任意字元
? 任意單個字元
[] 指定範圍內的任意單個字元
| 或者,如: a|b

範例:

#判斷輸入的是yes還是no

#!/bin/bash
read -p 'do you agree(yes|no): ' input

case $input in
y|yes)
	echo 'You input is yes'
	;;
n|no)
	echo 'you input is no'
	;;
*)
	echo 'you input is other'
	;;
esac
#運維菜單實現版本2

#!/bin/bash
cat <<EOF
請選擇:
1)備份資料庫
2)清理日誌
3)軟體升級
4)軟體回滾
5)刪庫跑路
EOF

read -p '請輸入上面的數字進行選擇: ' NUM

case $NUM in 
1)
	echo '備份資料庫'
	;;
2)
	echo '清理日誌'
	;;
3)
	echo '軟體升級'
	;;
4)
	echo '軟體回滾'
	;;
5)
	echo '刪庫跑路'
	;;
*)
	echo '以上所有'
	;;
esac

shell的迴圈語句:

  • for

  • while

  • until(不常用)

for迴圈格式

格式一:
for 變數 in 列表
do
	要做的操作
done
for列表的生成方式
  • 直接給出列表:1 2 3 4 5 ...

  • 整數列表:{start..end} 、$(seq [start [step]] end)

  • 通過$()命令替換得到列表

格式二:
for((表達式一;表達式二;表達式三));do
	要做的操作
done


範例
#使用for迴圈語句實現1+100
#!/bin/bash

sum=0
for num in `seq 100`;do   #列表1-100的表示方法; {1..100} seq 100
    let sum=$sum+$num  #let
done
echo $sum

#使用seq命令結合bc工具實現:
seq -s+ 100 | bc
-s:指定分隔符,預設分隔符是\n(換行)

#使用tr命令結合bc工具實現:
echo {1..100}|tr ' ' +|bc
#求100以內的奇數和
列表:{1..100..2} #一到100這個範圍,以2作為跳數
#99乘法表的實現
#!/bin/bash

for i in {1..9};do
    for j in {1..9};do
	 if [ $i -ge $j ];then
 	   echo -e "$i * $j = $[i*j]\t\c"  #\c很關鍵
         fi
    done
    echo
done

#\t:一個製表位,實現對齊的功能 \n:換行符 \c:有了-e參數,我們也可以使用轉義字元\c來強制 echo 命令不換行
#將指定目錄下文件的所有尾碼改名為 bak 尾碼
#將/data/下麵的所有文件該有bak尾碼
#!/bin/bash
DIR="/data/"
cd $DIR  || { echo "cannot inner this dir";exit; }
for FILE in * ;do #找到/data目錄裡面的文件列表   *是通配符:表示所有
    PRE=`echo ${FILE} | grep -Eo ".*\."`  #去掉文件的尾碼,如果沒有尾碼則不會匹配到
    sudo  mv ${FILE}  ${PRE}bakup #給去掉尾碼的文件名加上尾碼
done
#將目錄YYYY-MM-DD/中所有文件,移動到YYYY-MM/DD/下  每天都會生成一個以當天日期為名字目錄,裡面存放一些文件。

# 1. 創建模擬的實驗環境
#!/bin/bash
PDIR=/data/test
for i in {1..365};do 
 DIR=`date -d "-$i day" +%F` #-d:指定日期 -num:表示當前的前一天
 					#date -d "-1 year" +%F 表示以%F這種格式來顯示當前日期的前一天 
 mkdir -pv $PDIR/$DIR #創建指定的文件
 cd $PDIR/$DIR
 for j in {1..10};do
 touch $RANDOM.log  #隨機創建十個文件
 done
done

#將上面的目錄移動到YYYY-MM/DD/下
#!/bin/bash
DIR=/data/test #總目錄
cd $DIR || {  echo "無法進入 $DIR";exit; }
for subdir in * ; do  #*表示總目錄下麵的每一個子目錄的名稱
	YYYY_MM=`echo $subdir | cut -d"-" -f1,2` #取年月
	DD=`echo $subdir | cut -d"-" -f3`  #取日
	[ -d $YYYY_MM/$DD ] || mkdir -p $YYYY_MM/$DD &> /dev/null #判斷文件是否存在,不存在就創建一個
	mv $subdir/* $YYYY_MM/$DD #移動文件到新的文件夾下麵去
done
rm -rf $/DIR/*-*-* #刪除掉舊的文件


#date命令的使用
作用:顯示和設置系統時間
選項:
-d <字元串> 顯示字元串所指的日期與時間,比如:"-1 day" 表示當前日期的前一天,必須要加雙引號
-s <字元串> 設置當前的時間和日期
#時間日期格式:
%T:time,顯示時分秒,24小時制(hh:mm:ss)
%F:full date,顯示年月日,%Y-%m-%d

#範例:
使用date設置年月日:
date -s "20220710"或date -s "YYYY-MM-DD" #時間設置成2022年7月10日0時0分
只設置時間,不改年月日
date -s "hh:mm:ss"
設置全部時間
date -s "YYYY-YY-DD HH:mm:ss"

hwclock命令實現時間校對
-s, --hctosys #以硬體時鐘為準,校正系統時鐘
-w, --systohc #以系統時鐘為準,校正硬體時鐘

shell腳本的while循

while迴圈格式

while 判斷條件;do
      所作的操作
done

#迴圈控制條件;進入迴圈之前,先做一次判斷;每一次迴圈之後會再次做判斷;
#條件為“true”,則執行一次迴圈;直到條件測試狀態為“false”終止迴圈
#while迴圈實現1+...100
#!/bin/bash
sum=0
i=1
while [ $i -le 100 ] ;do
  let sum=$sum+$i
  let i+=1
done
echo $sum

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

-Advertisement-
Play Games
更多相關文章
  • Java面向對象(一) 一、面向過程(POP)與面向對象(OOP) 二者都是一種思想,面向對象是相對於面向過程而言的。面向過程,強調的是功能行為,以函數為最小單位,考慮怎麼做。面向對象,將功能封裝進對 象,強調具備了功能的對象,以類/對象為最小單位,考慮誰來做。 面向對象的三大特征: 封裝 繼承 多 ...
  • 稀疏組織 當一個數組中大部分元素為0,或者為同一個值的數組時,可以用稀疏數組來保存該數組 稀疏數組,記錄一共有幾行幾列,有多少個不同值 把具有不同值的元素和行里了及值記錄在一個小規模的數組中,從而縮小程式的規模! 我們定義一下原始數組: 原始數組如下: 0 0 3 0 0 0 0 0 0 4 0 0 ...
  • 歡迎關註公眾號:bin的技術小屋,本文圖片載入不出來的話可查看公眾號原文 本系列Netty源碼解析文章基於 4.1.56.Final版本 1. 前文回顧 在前邊的系列文章中,筆者為大家詳細剖析了 Reactor 模型在 netty 中的創建,啟動,運行,接收連接,接收數據,發送數據的完整流程,在詳細 ...
  • 一年一度的面試高峰期又來了,技術學習群的很多朋友問我有沒有關於JVM基礎面試題,網上各種面試題沒有分類很混亂,無法系統性參考學習。 於是,我就把之前整理的以及我面試過的真題和答案都整理了一份分享給大家。共計108道面試題! 持續根據技術群反饋不端更新,將涵蓋內容包括: Java設計模式、Spring ...
  • Arrays類的常用方法 代碼示例: public class ArrayDemo { public static void main(String[] args) { int[] a={1,2,57,8,1,58,4,51,5,45,15,1}; System.out.println(a); // ...
  • Java Development Kit 簡稱 JDK,任何需要開發 Java 程式的環境都需要進行安裝 JDK。 JDK 下載地址:https://www.oracle.com/java/technologies/downloads Windows 電腦安裝 JDK 1. 下載 JDK 在 JDK ...
  • 前言 .NET 7 的開發還剩下一個多月就要進入 RC,C# 11 的新特性和改進也即將敲定。在這個時間點上,不少新特性都已經實現完畢併合併入主分支 C# 11 包含的新特性和改進非常多,類型系統相比之前也有了很大的增強,在確保靜態類型安全的同時大幅提升了語言表達力。 那麼本文就按照方向從 5 個大 ...
  • 1.在C#中的目錄級別分為: 工程 -》項目-》類 。 2.在一個工程中可以添加多個項目,可以通過工程屬性設置運行時執行當前選中的項目: 選中當前工程 點擊 選擇屬性- 當前選定內容 3.在同一個工程中添加多個項目: 選擇當前工程-》添加-》 新建項目 4.移除同一個工程中的項目:(註意不要卸載項目 ...
一周排行
    -Advertisement-
    Play Games
  • 示例項目結構 在 Visual Studio 中創建一個 WinForms 應用程式後,項目結構如下所示: MyWinFormsApp/ │ ├───Properties/ │ └───Settings.settings │ ├───bin/ │ ├───Debug/ │ └───Release/ ...
  • [STAThread] 特性用於需要與 COM 組件交互的應用程式,尤其是依賴單線程模型(如 Windows Forms 應用程式)的組件。在 STA 模式下,線程擁有自己的消息迴圈,這對於處理用戶界面和某些 COM 組件是必要的。 [STAThread] static void Main(stri ...
  • 在WinForm中使用全局異常捕獲處理 在WinForm應用程式中,全局異常捕獲是確保程式穩定性的關鍵。通過在Program類的Main方法中設置全局異常處理,可以有效地捕獲並處理未預見的異常,從而避免程式崩潰。 註冊全局異常事件 [STAThread] static void Main() { / ...
  • 前言 給大家推薦一款開源的 Winform 控制項庫,可以幫助我們開發更加美觀、漂亮的 WinForm 界面。 項目介紹 SunnyUI.NET 是一個基於 .NET Framework 4.0+、.NET 6、.NET 7 和 .NET 8 的 WinForm 開源控制項庫,同時也提供了工具類庫、擴展 ...
  • 說明 該文章是屬於OverallAuth2.0系列文章,每周更新一篇該系列文章(從0到1完成系統開發)。 該系統文章,我會儘量說的非常詳細,做到不管新手、老手都能看懂。 說明:OverallAuth2.0 是一個簡單、易懂、功能強大的許可權+可視化流程管理系統。 有興趣的朋友,請關註我吧(*^▽^*) ...
  • 一、下載安裝 1.下載git 必須先下載並安裝git,再TortoiseGit下載安裝 git安裝參考教程:https://blog.csdn.net/mukes/article/details/115693833 2.TortoiseGit下載與安裝 TortoiseGit,Git客戶端,32/6 ...
  • 前言 在項目開發過程中,理解數據結構和演算法如同掌握蓋房子的秘訣。演算法不僅能幫助我們編寫高效、優質的代碼,還能解決項目中遇到的各種難題。 給大家推薦一個支持C#的開源免費、新手友好的數據結構與演算法入門教程:Hello演算法。 項目介紹 《Hello Algo》是一本開源免費、新手友好的數據結構與演算法入門 ...
  • 1.生成單個Proto.bat內容 @rem Copyright 2016, Google Inc. @rem All rights reserved. @rem @rem Redistribution and use in source and binary forms, with or with ...
  • 一:背景 1. 講故事 前段時間有位朋友找到我,說他的窗體程式在客戶這邊出現了卡死,讓我幫忙看下怎麼回事?dump也生成了,既然有dump了那就上 windbg 分析吧。 二:WinDbg 分析 1. 為什麼會卡死 窗體程式的卡死,入口門檻很低,後續往下分析就不一定了,不管怎麼說先用 !clrsta ...
  • 前言 人工智慧時代,人臉識別技術已成為安全驗證、身份識別和用戶交互的關鍵工具。 給大家推薦一款.NET 開源提供了強大的人臉識別 API,工具不僅易於集成,還具備高效處理能力。 本文將介紹一款如何利用這些API,為我們的項目添加智能識別的亮點。 項目介紹 GitHub 上擁有 1.2k 星標的 C# ...