Shell 编程规范与变量

1 Shell 脚本概述

在一些复杂的 Linux 维护工作中,大量重复性的输入和交互操作不仅费时费力,而且容 易出错,而编写一个恰到好处的 Shell 脚本程序,可以批量处理、自动化地完成一系列维护 任务,大大减轻管理员的负担。

1.1 Shell 的作用

Linux 系统中的 Shell 是一个特殊的应用程序,它介于操作系统内核与用户之间,充当 了一个“命令解释器”的角色,负责接收用户输入的操作指令(命令)并进行解释,将需要执 行的操作传递给内核执行,并输出执行结果。

[root@localhost ~]# cat /etc/shells
/bin/sh
/bin/bash
/sbin/nologin …… //省略部分内容

1.2 编写第一个 Shell 脚本

Bash(/bin/bash)是目前大多数 Linux 版本采用的默认 Shell。Bash 的全称为 Bourne Again Shell,是最受欢迎的开源软件项目之一。本课程中讲述的所有 Shell 操作,均以 Bash 为例。

[root@localhost ~]# vim first.sh //新建 first.sh 文件
cd /boot/
pwd
ls -lh vml*
[root@localhost ~]# chmod +x first.sh //添加可执行权限

上述 first.sh 脚本文件中,包括三条命令:cd /boot/、pwd、ls -lh vml*。执行此脚本文 件后,输出结果与依次单独执行这三条命令是相同的,从而实现了“批量处理”的自动化过程。

[root@localhost ~]# ./first.sh //直接运行脚本文件
/boot -rwxr-xr-x. 1 root root 5.2M Dec 10 2018 vmlinuz-0-rescue-0df49bdbc4fe42d4a24ed07e58d15436
-rwxr-xr-x. 1 root root 5.2M Nov 23 2016 vmlinuz-3.10.0-514.el7.x86_64

当然,一个合格的 Shell 脚本程序应该遵循标准的脚本结构,而且能够输出友好的提示 信息,更加容易读懂。对于代码较多,结构复杂的脚本,应添加必要的注释文字。

[root@localhost ~]# vim first.sh
#!/bin/bash
# This is my first Shell-Script. cd /boot
echo "当前的目录位于:" pwd
echo "其中以 vml 开头的文件包括:"
ls -lh vml*

上述 first.sh 脚本文件中,第一行“#!/bin/bash”是一行特殊的脚本声明,表示此行以后的 语句通过/bin/bash 程序来解释执行;其他以“#”开头的语句表示注释信息;echo 命令用于输 出字符串,以使脚本的输出信息更容易读懂。

1.3 重定向与管道操作

1.重定向操作

Linux 系统使用文件来描述各种硬件、设备等资源,如以前学过的硬盘和分区、光盘等 设备文件。用户通过操作系统处理信息的过程中,包括以下几类交互设备文件。

  • 标准输入(STDIN):默认的设备是键盘,文件编号为0,命令将从标准输入文件中读取在执行过程中需要的输入数据。
  • 标准输出(STDOUT):默认的设备是显示器,文件编号为1,命令将执行后的输出结果发送到标准输出文件。
  • 标准错误(STDERR):默认的设备是显示器,文件编号为2,命令将执行期间的各种错误信息发送到标准错误文件。

在实际的Linux系统维护中,可以改变输入、输出内容的方向,而不使用默认的标准输入、输出设备(键盘和显示器),这种操作称为重定向。

(1)重定向输出

重定向输出指的是将命令的正常输出结果保存到指定的文件中,而不是直接显示在显示器的屏幕上。重定向输出使用“>”或“>>”操作符号,分别用于覆盖或追加文件。

若重定向输出的目标文件不存在,则会新建该文件,然后将前面命令的输出结果保存到该文件中;若目标文件已经存在,则将输出结果覆盖或追加到文件中。

当需要保留目标文件原有的内容时,应改用“>>”操作符号,以便追加内容而不是全部覆盖。

(2)重定向输入

重定向输入指的是将命令中接收输入的途径由默认的键盘改为指定的文件,而不是等待从键盘输入。重定向输入使用“操作符。

通过重定向输入可以使一些交互式操作过程能够通过读取文件来完成。

[root@localhost ~]# vim pass.txt //添加初始密码串内容"123456"
123456
[root@localhost ~]# passwd --stdin jerry < pass.txt
//从 pass.txt 文件中取密码,需要注意 SELinux 会影响此命令执行,若执行失败可尝试关闭 SELinux
Changing password for user jerry.
 passwd: all authentication tokens updated successfully.

非交互式命令语句可以更方便的在 Shell 脚本中使用,从而大大减少程序被打断的过程, 提高脚本执行的效率。

(3)错误重定向

错误重定向指的是将执行命令过程中出现的错误信息(如选项或参数错误等)保存到指 定的文件,而不是直接显示在屏幕上。错误重定向使用“2>”操作符,其中“2”是指错误文件的 编号(在使用标准输出、标准输入重定向时,实际上省略了 1、0 编号)。

在实际应用中,错误重定向可用来收集程序执行的错误信息,为排错提供依据;对于 Shell 脚本,还可以将无关紧要的错误信息重定向到空文件/dev/null 中,以保持脚本输出的 简洁。

2.管道操作

管道(pipe)操作为不同命令之间的协同工作提供了一种机制,位于管道符号“|”左侧的 命令输出的结果,将作为右侧命令的输入(处理对象),同一行命令中可以使用多个管道。

在 Shell 脚本应用中,管道操作通常用来过滤所需要的关键信息。

重定向与管道操作是 Shell 环境中十分常用的功能,若能够熟练掌握并灵活运用,将有 助于编写代码简洁但功能强大的 Shell 脚本程序。

2 Shell 变量的作用、类型

2.1 自定义变量

自定义变量是由系统用户自己定义的变量,只在用户自己的 Shell 环境中有效,因此又 称为本地变量。在编写 Shell 脚本程序时,通常会设置一些特定的自定义变量,以适应程序 执行过程中的各种变化,满足不同的需要。

1.定义新的变量

Bash 中的变量操作相对比较简单,不像其他高级编程语言(如 C/C++、Java 等)那 么复杂。在定义一个新的变量时,一般不需要提前进行声明,而是直接指定变量名称并赋 给初始值(内容)即可。

定义变量的基本格式为“变量名=变量值”,等号两边没有空格。变量名称需以字母或下 划线开头,名称中不要包含特殊字符(如+、-、*、/、.、?、%、&、#等)。

2.查看和引用变量的值

通过在变量名称前添加前导符号“$”,可以引用一个变量的值。使用 echo 命令可以查看 变量,可以在一条 echo 命令中同时查看多个变量值。

[root@localhost ~]# echo $Product
Python
[root@localhost ~]# echo $Product $Version
Python 2.7.13

当变量名称容易和紧跟其后的其他字符相混淆时,需要添加大括号“{}”将其括起来,否 则将无法确定正确的变量名称。对于未定义的变量,将显示为空值。

root@localhost ~]# echo $Product2.5 //变量 Product2.5 并未定义
.5
[root@localhost ~]# echo ${Product}2.5
Python2.5

3.变量赋值的特殊操作

在等号“=”后边直接指定变量内容是为变量赋值的最基本方法,除此之外,还有一些特 殊的赋值操作,可以更灵活地为变量赋值,以便适用于各种复杂的管理任务。

(1)双引号(”)

双引号主要起界定字符串的作用,特别是当要赋值的内容中包含空格时,必须以双引号 括起来;其他情况下双引号通常可以省略。

在双引号范围内,使用“$”符号可以引用其他变量的值(变量引用),从而能够直接调 用现有变量的值来赋给新的变量。

(2)单引号(‘)

当要赋值的内容中包含$、“、\等具有特殊含义的字符时,应使用单引号括起来。在单 引号的范围内,将无法引用其他变量的值,任何字符均作为普通字符看待。但赋值内容中包 含单引号(‘)时,需使用\’符号进行转义,以免冲突。

[root@localhost ~]# PyVersion='Python $Version' //$符号不能再引用变量
[root@localhost ~]# echo $PyVersion //原样输出字符串
Python $Version

(3)反撇号(`)

反撇号主要用于命令替换,允许将执行某个命令的屏幕输出结果赋值给变量。反撇号括 起来的范围内必须是能够执行的命令行,否则将会出错。

需要注意的是,使用反撇号难以在一行命令中实现嵌套命令替换操作,这时可以改用 “$()”来代替反撇号操作,以解决嵌套的问题。

(4)read 命令

除了上述赋值操作以外,还可以使用 Bash 的内置命令 read 来给变量赋值。read 命令 用来提示用户输入信息,从而实现简单的交互过程。执行时将从标准输入设备(键盘)读入 一行内容,并以空格为分隔符,将读入的各字段依次赋值给指定的变量(多余的内容赋值给 最后一个变量)。若指定的变量只有一个,则将整行内容赋值给此变量。

为了使交互式操作的界面更加友好,提高易用性,read 命令可以结合“-p”选项来设置提 示信息,以便告知用户应该输入什么内容等相关事项。

4.设置变量的作用范围


默认情况下,新定义的变量只在当前的 Shell 环境中有效,因此称为局部变量。当进入子程序或新的子 Shell 环境时,局部变量将无法再使用。

[root@localhost ~]# echo "$Product $Version" //查看当前定义的变量值
Python 2.7.13
[root@localhost ~]# bash //进入子 Shell 环境
[root@localhost ~]# echo "$Product $Version" //无法调用父 Shell 环境中的变量
[root@localhost ~]# exit //返回原有的 Shell 环境

为了使用户定义的变量在所有的子 Shell 环境中能够继续使用,减少重复设置工作,可 以通过内部命令 export 将指定的变量导出为全局变量。用户可以同时指定多个变量名称作 为参数(无须使用“$”符号),变量名之间以空格分隔。

[root@localhost ~]# echo "$Product $Version" //查看当前定义的变量值
Python 2.7.13
[root@localhost ~]# export Product Version //将 Product、Version 设为全局变量
[root@localhost ~]# bash //进入子 Shell 环境
[root@localhost ~]# echo "$Product $Version" Python 2.7.13 //可以调用父 Shell 的全局变量
[root@localhost ~]# exit //返回原有的 Shell 环境

使用 export 导出全局变量的同时,也可以为变量进行赋值,这样在新定义全局变量时 就不需要提前进行赋值了。

5.数值变量的运算

Shell 变量的数值运算多用于脚本程序的过程控制(如循环次数、使用量比较等,后续 章节会介绍)。在 Bash Shell 环境中,只能进行简单的整数运算,不支持小数运算。整数 值的运算主要通过内部命令 expr 进行,基本格式如下所示。需要注意,运算符与变量之间 必须有至少一个空格。

expr 变量 1 运算符 变量 2 [运算符 变量 3]…

其中,变量 1、变量 2……对应为需要计算的数值变量(需要以“$”符号调用),常用的 几种运算符如下所述。

  • +:加法运算。
  • -:减法运算。
  • \*:乘法运算,注意不能仅使用“*”符号,否则将被当成文件通配符。
  • /:除法运算。
  • %:求模运算,又称为取余运算,用来计算数值相除后的余数。 以下操作设置了 X(值为 35)、Y(值为 16)两个变量,并依次演示了变量 X、Y 的 加、减、乘、除、取模运算结果。

若要将运算结果赋值给其他变量,可以结合命令替换操作(使用反撇号)。

2.2 特殊的 Shell 变量

1.环境变量

环境变量指的是出于运行需要而由 Linux 系统提前创建的一类变量,主要用于设置用户 的工作环境,包括用户宿主目录、命令查找路径、用户当前目录、登录终端等。环境变量的 值由 Linux 系统自动维护,会随着用户状态的改变而改变。

使用 env 命令可以查看到当前工作环境下的环境变量,对于常见的一些环境变量应了 解其各自的用途。

PATH 变量用于设置可执行程序的默认搜索路径,当仅指定文件名称来执行命令程序 时,Linux 系统将在 PATH 变量指定的目录范围查找对应的可执行文件,如果找不到则会提 示“command not found”。

在 Linux 系统中,环境变量的全局配置文件为/etc/profile,在此文件中定义的变量作用 于所有用户。除此之外,每个用户还有自己的独立配置文件(~/.bash_profile)。若要长期 变更或设置某个环境变量,应在上述文件中进行设置。

上述修改只有当 root 用户下次登录时才会生效。若希望立即生效,应手动修改环境变 量,或者可以加载配置文件执行。

[root@localhost ~]# history | wc -l
356 //已经记录的历史命令条数
[root@localhost ~]# source /root/.bash_profile //读取并执行文件中的设置
[root@localhost ~]# history | wc -l
200 //修改后的历史命令条数

2.只读变量

Shell 变量中有一种特殊情况,一经设定,其值是不可改变的,这种变量被称为只读变 量。在创建变量的时候可将其设置为只读属性,也可以将已存在的变量设置为只读属性,只 读变量主要用于变量值不允许被修改的情况。

3.位置变量

为了在使用 Shell 脚本程序时,方便通过命令行,为程序提供操作参数,Bash 引入了 位置变量的概念。当执行命令行操作时,第一个字段表示命令名或脚本程序名,其余的字符 串参数按照从左到右的顺序依次赋值给位置变量。

位置变量也称为位置参数,使用$1、$2、$3、…、$9 表示。

4.预定义变量

预定义变量是由 Bash 程序预先定义好的一类特殊变量,用户只能使用预定义变量,而 不能创建新的预定义变量,也不能直接为预定义变量赋值。预定义变量使用“$”符号和另一 个符号组合表示,较常用的几个预定义变量的含义如下。

$#:表示命令行中位置参数的个数。

$*:表示所有位置参数的内容。

$?:表示前一条命令执行后的返回状态,返回值为 0 表示执行正确,返回任何非 0 值均表示执行出现异常。关于$?变量的使用将在下一章介绍。

$0:表示当前执行的脚本或程序的名称。

  • 13
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值