shell脚本

一.什么是shell

Shell脚本语言是实现Linux/UNIX系统管理及自动化运维所必备的重要工具, Linux/UNIX系统的底层及基础应用软件的核心大都涉及Shell脚本的内容。Shell是一种编程语言, 它像其它编程语言如: C, Java, Python等一样也有变量/函数/运算符/if语句/循环控制/… 但在开始之前, 我想先理清Shell语言与Shell之间的关系.

当命令不在命令行中执行,而是从一个文件中执行时,该文件就是shell脚本。

Shell是一种解释型编程语言,不需要编译,执行时也是按行执行。

Shell脚本是由解释器解释执行的,常见的解释器有:bash dash ash ksh sh等

特点:

shell脚本是普通的文本文件,由流程控制逻辑和命令构成。

shell脚本通常以.sh作为后缀名,但不是必须的。

我们现阶段学习的主要是bash dash.

1.什么是Shell脚本

命令、变量和流程控制语句等有机的结合起来

shell脚本擅长处理纯文本类型的数据,而linux中,几乎所有的配置文件,日志,都是纯文本类型文件

2.脚本语言的种类

1、编译型语言

定义:指用专用的编译器,针对特定的操作平台(操作系统)将某种高级语言源代码一次性翻译成可被硬件平台直接运行的二进制机器码(具有操作数,指令、及相应的格式),这个过程叫做编译(./configure make makeinstall );编译好的可执行性文件(.exe),可在相对应的平台上运行(移植性差,但运行效率高)。。

典型的编译型语言有, C语言、C++等。

另外,Java语言是一门很特殊的语言,Java程序需要进行编译步骤,但并不会生成特定平台的二进制机器码,它编译后生成的是一种与平台无关的字节码文件(*.class)(移植性好的原因),这种字节码自然不能被平台直接执行,运行时需要由解释器解释成相应平台的二进制机器码文件;大多数人认为Java是一种编译型语言,但我们说Java即是编译型语言,也是解释型语言也并没有错。

2**、解释型语言**

定义:指用专门解释器对源程序逐行解释成特定平台的机器码并立即执行的语言;相当于把编译型语言的编译链接过程混到一起同时完成的。

解释型语言执行效率较低,且不能脱离解释器运行,但它的跨平台型比较容易,只需提供特定解释器即可。

常见的解释型语言有, Python(同时是脚本语言)与Ruby等。

3**、脚本语言**

定义:为了缩短传统的编写-编译-链接-运行(edit-compile-link-run)过程而创建的计算机编程语言。

特点:程序代码即是最终的执行文件,只是这个过程需要解释器的参与,所以说脚本语言与解释型语言有很大的联系。脚本语言通常是被解释执行的,而且程序是文本文件。

典型的脚本语言有,JavaScript,Python,shell等。

其他常用的脚本语句种类

shell脚本的优势在于处理操作系统底层的业务 (linux系统内部的应用都是shell脚本完成)因为有大量的linux系统命令为它做支撑。2000多个命令都是shell脚本编程的有力支撑,特别是grep、awk、sed等。例如:一键软件安装、优化、监控报警脚本,常规的业务应用,shell开发更简单快速,符合运维的简单、易用、高效原则.

shell解释器,用户和操作系统内核之间的桥梁

shell介于操作系统内核与用户之间,负责接收用户输入的操作指令(命令),并运行和解释,将需要执行的操作传递给操作系统内核并执行

shell程序在系统中充当了一个”命令解释“的角色

4.1 Shell常见种类

Bsh:由贝尔实验室编写。Bsh是产生较早的UNIX Shell程序,实现了最基本的命令解释器的功能,同时也可以作为脚本编程语言。

Csh:是因使用C语言的语法风格而得名,在用户的命令行交互界面上进行了很多改进,并增加了历史,别名,文件名替换,作业掏等功能,相比Bsh,Csh在更加适用为  用户提供命令交互操作。

Ksh:在Bsh和Csh之后出现的,结合了两都的功能优势,兼具Bsh的语法和Csh的交互特性。

Bash:从名称可以看出是Bsh的升级版本,是著名的开源软件项目,目前大多数的Linux版本(包括Red Hat公司的Linux系统)都使用Bash 作为默认的Shell程序当运行Shell程序时,实际运行的是Bash程序。

Zsh:更多地基于交互式操作进行设计的Shell程序,集成了Bash,Ksh等多种Shell程序的优点。

Linux默认shell是Bourne Again shell(bash)

[root@localhost /]# cat /etc/shells

 3、书写规范
                脚本文件尽量放在一个文件夹下面,尽量名字见名知义,方便查找

                以 .sh 为后缀

                优秀的编码习惯 [ ] 

                = 等号两端不能有空格,不然赋不上值

       4、单引号&双引号
                数字:

                单引号、双引号、或者是不加,结果是一样的 都是数字字符串

                单引号和双引号是一样的。

                取值:

                单引号是原文输出,双引号是取值

5、 bash版本

                查看bash的version

                /bin/bash --version

  5、 sh与bash 的关系:
                sh是一种POSIX标准,它有很多种实现,包括ksh88, dash,bash等。

                因为sh是一种规范,并不是实现,

                所以/bin/sh实际上是一个硬链接,链接到某种实现上。大多数情况下,

                /bin/sh会链接到/bin/bash。所以执行sh xx.sh

                等价于执行  bash xx.sh

二、脚本解释器
        1、 选择解释器
                开头的"#!"字符又称为幻数,在执行bash脚本的时候,内核会根据"#!"后的解释器来确定该用那个程序解释这个脚本中的内容。

                init.d目录。这个目录是干嘛的呢?

                它归根结底只做了一件事情,但这件事情非同小可,是为整个系统做的,

                因此它非常重要。init.d目录包含许多系统各种服务的启动和停止脚本。

                [root@localhost ~]# head -1 /etc/init.d/*

                ==> /etc/init.d/functions <==

                # -*-Shell-script-*-

                ==> /etc/init.d/netconsole <==

                #!/bin/bash

                ==> /etc/init.d/network <==

                #! /bin/bash

                ==> /etc/init.d/README <==

                You are looking for the traditional init scripts in /etc/rc.d/init.d,

                init.d目录都是用来放服务脚本的,当Linux启动时,会寻找这些目录中的服务脚本,并根据脚本的run level确定不同的启动级别。

        2、 开发的规范
                1) 放在统一的目录

                2) 脚本以.sh为扩展名 (不是必需的)

                3) 开头指定脚本解释器。(#! /bin/sh)

                4) 开头加版本版权等信息,可配置~/.vimrc文件自动添加。

                5) 脚本不要用中文注释,尽量用英文注释。

                6) 代码书写优秀习惯

                a、成对的内容一次性写出来,防止遗漏,如[ ]、' '、" "等

                b、[ ] 两端要有空格,先输入[ ],退格,输入2个空格,再退格写。

                c、流程控制语句一次书写完,再添加内容。(if 条件 ; then 内容;fi)

                语法结构为

                        if condtion

                        then

                        do something

                        elif condtion

                        then

                        do something

                        else

                        do something

                        fi

                d、通过缩进让代码易读。

                f、脚本中的引号都是英文状态下的引号,其他字符也是英文状态。

三、脚本的编写/执行/调试        
        1、 创建第一个shell脚本文件
                创建脚本文件test.sh

                [root@localhost tmp]# mkdir demo1

                [root@localhost tmp]# cd demo1

                [root@localhost demo1]# ls

                [root@localhost demo1]# touch test.sh

                [root@localhost demo1]#

                在shell编程中,通常情况下,#代表注释,但是第一行的#是一个特例。

                #! /bin/sh是shell脚本的一个标志,声明这个script使用的shell。

                第一行的#!是一个约定标记, 它告诉脚本这段脚本需要什么解释器来执行.

                第二行的echo命令则负责向屏幕上输出一句话。

                运行这个脚本文件 sh test.sh 或者是 ./test.sh

刚开始执行不成功,是因为权限不够 如果赋权限 使用chmod命令

                报错原因:

                        根据截图也可以看出,text01.sh的文字颜色是白色,并不是绿色,因此我们可以赋权限, 可以使用chmod 777 test.sh命令来赋权限。

其实运行shell程序共有三种方法,除了给文件赋予可执行权限外,还有另外两种方法。

chmod +x使文件具有可执行权限, 直接运行;
直接调用解释器, 将脚本文件作为参数传入 (比如bash test.sh)
使用source(也可用.代替)执行文件 (在当前bash环境下读取并执行FileName中的命令。该filename文件可以无"执行权限")

通常情况下, 最方便的方式就是方式1, 通过方式1执行你需要在脚本第一行写好这段脚本由哪个解释器来解释, 而通过方式2来执行则没有这个限制, 写了也没用。除此之外方式1与方式2执行命令就没有区别了,

                source script.sh:在脚本运行结束后,脚本中的变量在当前环境仍会被保留。

                sh script.sh:在当前环境启动一个子进程运行脚本, 脚本中的变量会在脚本运行结束时释放掉。

        文件权限解读

          开头的-rwxrw-r--这一字符串标识文件权限。

          这个字符串有10位,可以分为4段来解读。注:r--可读,w--可写,x--可执行。

          第一段(第1位)表示是目录还是文件,-表示是文件,d表示是目录;

          第二段(第2-4位,共3个字符串)表示文件所属用户对它的权限;

          第三段(第5-7位,共3个字符串)表示文件所属用户组用户对它的权限;

          第四段(第8-10位,共3个字符串)表示其他用户对它的权限;

                读取权限 r = 4

                写入权限 w = 2

                执行权限 x = 1

                775 这三个数字代表拥有者,组用户,其他用户的权限。

                例如:

                        7 拥有者有 读取,写入,执行权限

                        7 组用户有 读取,写入,执行权限

                        5 其他用户有 读取,执行权限(4+1 = 5)

                777 与 775的区别是

                        其他用户有写入权限,而775的没有。

                        举个例子:

                        文件A,权限是775

                        root是拥有者

                        www-data是组用户

                        ooo 是其他用户

                那么

                        root,www-data,ooo都能对文件A读取和执行。

                        root,www-data 能对文件A写入。

                        ooo不能对文件A写入。

                        如果是777

                        则三个用户都能读取,写入,执行文件A

        2、 Shell的变量
                变量可以分为三类:环境变量(全局变量)、普通变量(局部变量)、 特殊变量

                环境变量:也可称为全局变量,可以在创建他们的Shell及其派生出来的任意子进程shell中使用,环境变量又可分为自定义环境变量和Bash**内置的环境变量**

                普通变量:也可称为局部变量,只能在创建他们的Shell函数或Shell脚本中使用。普通变量一般是由开发者用户开发脚本程序时创建的。

                特殊变量:脚本内置的具有特殊用途的变量

                使用 env ,export 命令查看系统中的环境变量

                env 显示用户的环境变量

                export 显示当前导出成用户变量的shell变量,并显示变量的属性(是否只读),

                按变量名称排序

                输出一个系统中的 环境变量 echo $HOME

                2.1 普通变量
                        本地变量在用户当前的Shell生存期的脚本中使用。例如,本地变量a取值为1,这个值在用户当前Shell生存期中有意义。如果在Shell中启动另一个进程或退出,本地变量值将无效。

 注意:$用来获取变量的值 ,=前后两端不能有空格

#! /bin/sh
#输出数字,字符串
echo "不带引号的数字:"
echo 1
echo "带单引号的数字:"
echo '1'
echo '双引号数字:'
echo "1"
echo "带单引号字符串:"
echo 'hello'
echo "双引号字符串:"
echo "hello"
#定义变量
str="123"
#输出变量
echo $str
echo "单引号变量:"
echo '$str'
echo '双引号变量:'
echo "$str"
#定义转义字符
echo '转义字符\'\'
echo "转义字符\"\""

注意:

                单引号字符串的限制:单引号里的任何字符都会原样输出,单引号字符串中的变量是无效的;

                单引号字串中不能出现单独一个的单引号(对单引号使用转义符后也不行),但可成对出现,作为字符串拼接使用。

                双引号里可以有变量,可以出现转义字符。

                     2.2 变量中引号的使用
                                只有在变量的值中有空格的时候,会使用引号。

                        单引号与双引号的区别在于,是否能够解析特殊符号。

                        变量使用反引号赋值,及使用${}获取参数值

 取值的时候不加{}代表取的是某一个变量的值 :$ab 取的是ab变量的值

                加{}之后取的是拼接之后的值:${a}b 取的是 ${a}的值拼接上字符串b

                2.3 定义变量名技巧
变量名只能为字母、数字或下划线,只能以字母或下划线开头。
变量名的定义要有一定的规范,并且要见名知意。
                示例:

                ClsnAge=22 #<==每个单词的首字母大写的写法

                clsn_age=22 #<==单词之间用"_"的写法

                clsnAgeSex=man #<==驼峰语法:首个单词的首字母小写,其余单词首字母大写

                CLSNAGE=22 #<==单词全大写的写法

一般的变量定义、赋值常用双引号;简单连续的字符串可以不加引号;希望原样输出时使用单引号。
希望变量的内容是命令的解析结果时,要用反引号``,或者用$()把命令括起来再赋值。
        

                2.4 特殊变量
位置变量
                常用的特殊位置参数说明

位置变量

作用说明

$0

获取当前执行的shell脚本的文件名,如果执行脚本带路径那么就包括脚本路径。

$n

获取当前执行的shell脚本的第n个参数值,n=1..9,当n为0时表示脚本的文件名,如果n大于9用大括号括起来{10},参数以空格隔开。

$#

获取当前执行的shell脚本后面接的参数的总个数

$*

获取当前shell的所有传参的参数,不加引号同$@

(将接收到的每一个参数当做每一份数据,每个参数之间用空格来分开)

加上引号表示将传入的多个参数从整体上当做一份数据,以"1 2…n"的形式输出所有参数。

$@

获取当前shell的所有传参的参数,不加引号同$*

加上引号

表示仍然将传入的多个参数当做多份数据,空格区分,彼此之间独立,以"1" "2"…"n" 的形式输出所有参数。

当“$*”和“$@”都加双引号时,两者有区别,都不加双引号时,两者无区别。
 

2. 进程状态变量

Shell进程的特殊状态变量说明

1. echo**参数说明**

  3、定义变量的方式
                3.1三种定义变量的方式
直接赋值 a=1
传参 (传递参数)
交互式设置变量,使用read命令 类似于Scanner
                3.2 read命令说明
                获取输入内容,在命令行中使用

                [root@localhost demo1]# read

                123

                [root@localhost demo1]# echo $REPLY

                123

                [root@localhost demo1]# read str

                1234

                [root@localhost demo1]# echo $str

                1234

                [root@localhost demo1]# echo $?

                0

                [root@localhost demo1]#

        4、 变量的子串
               4.1变量子串说明

#!/bin/sh

                str="1bcabdbdbdffgjrjhrhgotababab"

                echo "输出字符串:${str}"

                echo "字符的长度为:${#str}"

                echo "从第三位开始截取字符串:${str:3}"

                echo "从第三位开始截取6位:${str:3:6}"

                echo "从头开始删除最短匹配的ab:${str#ab}"

                echo "从头开始删除最长匹配的ab:${str##ab}"

                echo "从尾部开始删除最短匹配的ab:${str%ab}"

                echo "从尾部开始删除最长匹配的ab:${str%%ab}"

                echo "替换第一个ab:${str/ab/xxx}"

                echo "替换所有的ab:${str//ab/xxx}"

    4.2 特殊扩展变量说明

四、条件表达式
        1、 &&,||
                如果第一个命令执行成功(返回1),与操作符&&才会执行第二个命令

                如果第一个命令执行不成功就直接返回0

                如果第一个命令执行失败,或操作符||才会执行第二个命令

                如果第一个命令执行成功就直接返回1

  2、文件判断

                常用文件测试操作符

   判断目录是否存在

   3、字符串判断

                字符串测试操作符

  4、 整数判断

                整数二元比较操作符参考

                [[ == ]]

                -eq []

 5、 逻辑符号
                &&:连接命令,如果第一个命令执行成功(返回1),与操作符&&才会执行第二个命令

                [root@localhost demo03]# expr 1 == 1 && echo "相等"

                1

                相等

                [root@localhost demo03]# expr 1 == 2 && echo "相等"

                0

                [root@localhost demo03]# expr 1 == 1 && echo "相等"

                1

                &&:[[]]使用,作为逻辑运算符,判断两个表达式是否都成立

                [root@localhost demo03]# [[ 1 == 1 && 1 == 1 ]]

                [root@localhost demo03]# echo $?

                0

                [root@localhost demo03]# [[ 1 == 1 && 1 == 2 ]]

                [root@localhost demo03]# echo $?

                1

                常用逻辑操作符
 

五、 if条件语句

        1、单分支语句

                if 条件表达式

                then

                Do something

                fi

 2、双分支语句

                if 条件表达式

                then

                。。。。。。

                else

                fi

   3、多分支语句
                if 条件

                then

                elif

                then

                else

                fi
 

>=90优秀
>=80 良好
>=70 中等
>=60 及格
#! /bin/sh
if [ $# == 0 ]
then
 echo "没有参数"
else
    if [ $1 -ge 90 ]
     then
       echo "非常优秀!!!"
    elif [ $1 -ge 80 ]
      then
        echo "优秀"
    elif [ $1 -ge 70 ]
       then
        echo "良好"
    elif [ $1 -ge 60 ]
     then
      echo "及格"
    else
      echo "不及格"
    fi
fi

六、 case条件结构语句

        1、 case语法结构

case "变量" in
值1)
;;
值2)
;;
*)
;;
esac
#!/bin/bash
case $1 in
   1)
    echo "第一季度"
    ;;
   2)
    echo "第二季度"
    ;;
   *)
    echo "其他季度" 
 
esac

      2、 case与if的对比

	case书写方式
case $name in
  值1) 
      指令1
      ;;
  值2) 
      指令2
      ;;
   *) 
      指令
esac
 
 
	if书写方式
if [ $name == "值1" ]
  then 
    指令1
elif [ $name == "值2" ]
  then 
    指令2
else
    指令    
fi

  3、 case值的书写方式

apple)

echo -e "$RED_COLOR apple $RES"

;;

也可以这样写,输入2种格式找同一个选项

apple|APPLE)

echo -e "$RED_COLOR apple $RES"

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值