Shell:一般我们是用图形界面和命令去控制计算机,真正能够控制计算机硬件(CPU、内存、显示器等)的只有操作系统内核(Kernel),由于安全、复杂、繁琐等原因,用户不能直接接触内核,需要另外再开发一个程序,让用户直接使用这个程序;该程序的作用就是接收用户的操作(点击图标、输入命令),并进行简单的处理,然后再传递给内核,内核和用户之间就多了一层“中间代理”,Shell 其实就是一种脚本语言,也是一个可以用来连接内核和用户的软件,我们编写完源码后不用编译,直接运行源码即可。

常用的Shell:bash由 GNU 组织开发,sh 是 UNIX 上的标准 shell,是第一个流行的 Shell,bash保持了对 sh shell 的兼容性,是各种 Linux 发行版默认配置的 shell。现在sh 已经基本被 bash 代替,bash是sh的扩展补充,但是也有些是不兼容的,大多数情况下区别不大,特殊场景可以使用 bash 代替 sh。

img

在ubuntu下,上图是我们打开的终端,这里输入的命令就是我们shell的命令,一般$表示的是普通用户,而#表示的是超级用户(root)

1、第一个脚本程序:

在终端创建一个.sh文件,vi test.sh

img

第一行,#!是一个约定的标记,它告诉系统这个脚本需要什么解释器来执行,用的是哪种shell,后面的/bin/bash就是指明了解释器的具体位置。
第二行#是注释行,用来解释说明,当然#!是特殊的,不在此类。 第三行在是终端输出 hello,is me!

img

执行shell脚本:

./test.sh: ./表示当前目录,执行./test.sh会说明权限不够,不能执行。需要改变文件的权限:chmod 777 test.sh,就能执行

. ./test.sh: 在不想改变权限的时候,测试脚本是不是能够正常使用。. 临时增加权限。

2、shell获取字符串长度:

#!/bin/bash
#获取字符串长度
str="123456"
echo "${#str}"

结果:

img

3、shell变量:

在bash shell中,每一个变量的值都是字符串, 当然也可以用declare 关键字显式定义变量的类型,在赋值的时候等号两边不能有空格,如:str=1 ,str=‘1’ ,str=“1”,变量名必须有字母、下划线、数字组成,开头必须字母或者下划线,不能用shell。

  • 局部变量:shell也有自定义函数,函数里面的变量为局部变量,但是它也是相当于全局变量,函数中的变量,在函数外调用也是可以的,如果要仅限函数使用,需要在函数变量前加个关键字:local
  • 全局变量:每打开一个终端就是一个shell会话,在这个shell会话(终端)定义的变量就是全局变量,它在这个shell会话有效,当你打开另一个终端就是另一个shell会话,这个变量在另一个终端就失效了。
  • 环境变量:在全局变量前加 export ,如:export a=1 那么这个变量就是环境变量了。创建这个变量的shell成为父shell,这个shell中,在创建一个shell叫做子shell,环境变量可以由父shell往下一级一级传,而不能逆转往上传递。当shell会话销毁时,这个环境变量也会随之销毁。想要永久保存就得环境变量写到启动文件中去。

4、引用shell变量

使用shell变量在变量前面加一个$,而标准的是 &{},目的是在一长串字符中可以识别出这个变量,而不会引起误会,如下:

#!/bin/bash
#引用shell变量
str="abc"
 
echo "$str"
echo "the str vaile is: $str1" 
#后面增加一个1,就不能正确的识别变量
echo "the str vaile is: ${str}1"
#所以我们引用变量,最好统一用 ${} 的形式

结果:

img

5、shell变量的赋值、修改、删除:

a)、shell变量的赋值

#!/bin/bash
#变量的赋值
n=1
v1=${n}
v2='${n}'
v3="${n}"
 
echo "${v1}"
echo "${v2}"
echo "${v3}"
 

img

可以从结果看出不加引号和加双引号的结果是相同的,而单引号是原样输出变量后面赋值的内容。

b、shell变量的修改、删除:

#!/bin/bash
#变量值的修改
a=1
echo "a: ${a}"
a=2
echo "a: ${a}"
 
#只读变量是不可以修改的,在变量前加readonly,就是只读变量
c=1
echo "c: ${c}"
readonly c
c=2
echo "c: ${c}"
 
#只要在变量前面加一个 unset,如: unset a   就可以删除变量。
unset a
echo "a: ${a}"

img

5、shell特殊变量:

$0当前脚本的文件名或者解释器。
$n(n≥1)传递给脚本或函数的参数。n 是一个数字,表示第几个参数。例如,第一个参数是 $1,第二个参数是 $2。
$#传递给脚本或函数的参数个数。
$*传递给脚本或函数的所有参数。
$@传递给脚本或函数的所有参数。当被双引号" "包含时,$@ 与 ∗ 稍 有 不 同 , * 稍有不同, *的所有参数是一个数据,而$@一个参数就是一份数据
$?上个命令的退出状态,或函数的返回值
$$当前 Shell 进程 ID。对于 Shell 脚本,就是这些脚本所在的进程 ID。

**脚本:**如果参数个数太多,达到或者超过了 10 个,那么就得用${n}的形式来接收了,例如 ${10}${11}

#!/bin/bash
#shell特殊变量
#$n: $0表示本脚本,$1表示输入的第一个参数:11,$2表示输入的第一个参数:22    
echo "$ 0: $0"
echo "a: $1"
echo "b: $2"
#$*: 输入的参数 11 22
echo "$ *: $*"
#$@: 输入的参数 11 22
echo "$ @: $@"
#$$: shell进程id
echo "$ $: $$"
#$#: 参数个数 2
echo "$ #: $#"

img

函数:如果参数个数太多,达到或者超过了 10 个,那么就得用${n}的形式来接收了,例如 ${10}${11}

#!/bin/bash
#shell特殊变量
#$n: $0表示本脚本,$1表示输入的第一个参数:11,$2表示输入的第一个参数:22    
function temp(){
    echo "$ 0: $0"
    echo "a: $1"
    echo "b: $2"
    #$*: 输入的参数 11 22
    echo "$ *: $*"
    #$@: 输入的参数 11 22
    echo "$ @: $@"
    #$$: shell进程id
    echo "$ $: $$"
    #$#: 参数个数 2
    echo "$ #: $#"
 }
 
 temp 11 12

img

$?:上一个命令的退出状态,或者返回值:

#!/bin/bash
 
if [ $1 == 11 ]
then
    return 0
else
    return 1
fi

img

$?在shell函数中的应用:

#!/bin/bash
function temp(){
    return $1
}
temp 3
 
#$?: 这里是函数的返回值
echo "$ ?: $?"

img

6、shell中字符串的拼接:

#!/bin/bash
#shell字符串的拼接
n1=ab
n2=cd
 
temp1=${n1}${n2}  #中间不能有空格 
temp2="${n1}${n2}"
temp3="${n1}cdefg"
 
 
echo "${temp1}"
echo "${temp2}"
echo "${temp3}"

img

7、字符串的截取

  • ${string: start :length}:字符串从左边start开始的位置截取length个字符,如果没有length,就从左边start开始到结束
  • ${string: 0-start :length}:字符串从右边start开始的位置截取length个字符,如果没有length,就从右边数start开始到结束
  • ${string#*chars}:字符串从边开始的第一个chars开始的位置截取到边结束,chars本身不在截取范围内
  • ${string##*chars}:字符串从边开始的最后一个chars开始的位置截取到边结束,chars本身不在截取范围内
  • ${string%*chars}:字符串从边边开始的第一个chars开始的位置截取到边结束,chars本身不在截取范围内
  • ${string%%*chars}:字符串从边开始的最后一个chars开始的位置截取到边结束,chars本身不在截取范围内
#!/bin/bash
#shell截取字符串
 
str=http//:baidu.com/.c
 
echo "length: ${str:7:5}"   # baidu
echo "no length: ${str:7}"  # baidu.com/.c
 
echo "length: ${str:0-7:5}"  # .com/
echo "no length: ${str:0-7}" # .com/.c
 
echo "#: ${str#*/}"   # /:baidu.com/.c
echo "##: ${str##*/}" # .c
 
echo "%: ${str%/*}"   # http//:baidu.com
echo "%%: ${str%%/*}" # http

img

8、shell中的数组:

bash shell 中只支持一维,而不支持二维,定义的形式:array=(n1 n2 n3) ,数组名等号两边不能有空格,数组变量的值用空格隔开表示不同的值,一个数组变量的值可以用数字或者字符串不同的形式组成:array=(1 2 ab) 。

Shell 数组的长度不是固定的,定义之后还可以增加元素,如:array[3]=cd ;就在最后增加了一个元素。

也不用逐个赋值;如:array=([2]=ab);但他的长度是1

调用数组:

  • ${array[0]}
  • a=${array[0]}
  • ${array[*]} //表示数组的所有元素
  • ${array[@]} //表示数组的所有元素
#!/bin/bash
#shell中的数组
array=( 1 2 ab)
 
n=${array}  #array中的第一个元素赋值给n
 
echo "array[0]: ${array[0]}"
echo "array[*]: ${array[*]}"
echo "n[*]: ${n[*]}"
 
array[5]=cd
echo "array[5]: ${array[5]}"
echo "array[@]: ${array[@]}"

img

获取shell数组的元素个数或者shell数组里字符串的长度:${#array[*]}、${#array[@]}、${#array[4]}

#!/bin/bash
#shell中的数组
array=( 1 2 abc)
 
echo "#array[*] ${#array[*]}"
 
array[4]=12345
echo "#array[@] ${#array[*]}" #这边多加了一个元素就是4
echo "#array[4] ${#array[4]}" #这边是计算array[4]中字符串的长度

img

数组的拼接:利用${array[*]}、${array[@]}

#!/bin/bash
#shell中的数组的拼接
array1=( 1 2 abc)
array2=(cd ef)
array3=(${array1[@]} ${array2[@]})
 
echo "array3: ${array3[@]}"
echo "array3 length: ${#array3[@]}"

img

shell数组元素的删除:unset

#!/bin/bash
#shell中的数组
array1=(1 2 abc)
array2=(1 2 abc)
 
unset array1[2]  #删除array1数组中的一个元素abc
echo "array1[*]: ${array1[*]}"
 
 
unset array2  #删除整个数组array2
echo "array2[*]: ${array2[*]}"

img

9、shell中条件判断if:

a)、if单分支:

-eq 等于,如:if [ “$a” -eq “$b” ]

-ne 不等于,如:if [ “$a” -ne “$b” ]

-gt 大于,如:if [ “$a” -gt “$b” ]

-ge 大于等于,如:if [ “$a” -ge “$b” ]

-lt 小于,如:if [ “$a” -lt “$b” ]

-le 小于等于,如:if [ “$a” -le “$b” ]

#!/bin/bash
#shell中if:
#if单分支第一中形式:if []
if [ 1 -eq 1 ] #[]里面的数据和中括号必须一个空格 
then
        echo "first if:真"
fi
 
#if单分支第二中形式:if []; then
if [ 1 -eq 1 ]; then #then和if写在同一行必须要用; 
        echo "second if:真"
fi
 
#if单分支第三中形式:if (())
if ((1==1))
then
        echo "third if:真"
fi

img

b、双分支if:

#!/bin/bash
#shell中if:
#if双分支
if ((1==0)) #判断1和o是不是相等
then
        echo "我是真的"
else
        echo "我是假的"
fi
 
#................................
if [ 1 -eq 0 ] #判断1和o是不是相等,
then
        echo "我是真的"
else
        echo "我是假的"
fi

img

3、iif多分支:

#!/bin/bash
#shell中if:
#if多分支
if ((1==0)) #判断1和o是不是相等
then
        echo "1=0;是真的"
elif ((1>0))
then
        echo "1>0;是真的"
else
        echo "1>0;是假的"
fi
 
#..............................
if [ 1 -eq 0 ] #判断1和o是不是相等
then
        echo "1=0;是真的"
elif [ 1 -gt 0 ]
then
        echo "1>0;是真的"
else
        echo "1>0;是假的"
fi

img

10、shell中的 case语句:

#!/bin/bash
#shell中的case
 
read VAILE #在屏幕中输入一个数,1或者2,如果不是这两个数就代表其他;即 *),
case ${VAILE} in
        "1")
                echo "我是1"
        ;;
 
        "2")
                echo "我是2"
        ;;
 
        *)
                echo "我不是1,也不是2"
        ;;
esac

img

11、shell中的 for循环:

a)、for循环形式1:

#!/bin/bash
#shell中的 for
for n in 1 3 5  #n是我们自定义的变量,in后面三个数就是循环3次,每次的值从第一个数的值开始
do
        echo "我是n,我的值是:${n}"
done

img

b)、for循环形式2:与c语言的形式类似

#!/bin/bash
#shell中的 for
for ((n=0;n<3;n++))
do
        echo "我是n,我的值是:${n}"
done

img

12、shell中的 while循环:

#!/bin/bash
#shell中的 while
n=0
while [ ${n} -lt 2 ] #n<2;就循环,否则退出循环
do
        n=$((${n}+1))
        echo "我是n,我的值是:${n}"
done
        echo "我退出循环了"

img

13、shell中的 until循环:

#!/bin/bash
#shell中的 until
n=-2
until ((n>1)) #不满足条件就循环,满足条件则退出循环
do
        n=$((${n}+1))
        echo "我是n,我的值是:${n}"
done
        echo "我退出循环了"
 

img

14、shell循环中的break和continue:

#!/bin/bash
#shell中的 break/continue
n=0
while ((n<6)) #n<6;就循环,否则退出循环
do
        n=$((${n}+1))
        if ((n==2))
        then
                echo "我应该打印2,但我continue了"
                continue
        fi
 
        if ((n==5))
        then
                echo "n循环6次,应该等于6,但我break了,我现在的值是:${n}"
                break
        fi
        echo "我是n,我的值是:${n}"
 
done
        echo "我退出循环了"
 

img

15、shell中的函数:

function 函数名(){

}

#!/bin/bash
#shell中的函数
function fun(){
        if(($1 > 1)); then
                echo "条件成立"
        fi
 
        if(($2 > 1)); then
                echo "条件成立"
        else
                echo "条件不成立"
        fi
}
 
fun 2 1   #这边传参数给fun()函数
 

img

Logo

为开发者提供学习成长、分享交流、生态实践、资源工具等服务,帮助开发者快速成长。

更多推荐