Linux Bash Set命令解析

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
云解析 DNS,旗舰版 1个月
全局流量管理 GTM,标准版 1个月
简介: 平时在学习大牛的Shell脚本时,我们经常在脚本的开头看到很多set开头的命令

概述


平时在学习大牛的Shell脚本时,我们经常在脚本的开头看到很多set开头的命令,比如


#! /usr/bin/env bash
set -e
set -u
set -x
或者
set -eux


但是,人们经常忽略这几个set命令的含义,我要说的是这个命令的作用其实非常的强大,可以提供脚本的debug效率和安全性。好了,下面我们就一 一分析一下这几个命令的真正含义。


set命令为shell内建命令,通过help set可以看到关于set的帮助信息。其主要作用是改变 shell 选项和位置参数的值,或者显示 shell 变量的名称和值。在终端下,如果执行set命令,就会显示当前shell下所有的环境配置信息。


错误处理


一般情况下,每个Linux shell命令执行完毕后,都会返回一个执行结果,并将其保存到$?中,一般0表示命令执行成功,非零表示命令执行失败。比如,


$ set
$ echo $?
0           
0表示set命令执行成功


shell脚本中一般会使用到大量的命令,严格意义上,我们应该小心的检查每个关键命令的执行结果,以确定之后的执行逻辑。一般的处理情形如下


#! /usr/bin/env bash
PID=`pidof dockers`
RET=$?
if [ $RET -eq 0 ];then
  echo "docker's pid is $PID"
else
  echo "docker don't run"
fi 
 #这里故意写成dockers,这样pidof就会执行失败。


如果命令较少的话,这种写法可能还能被接受,如果shell脚本的规模比较大时,可想而知,如果都按照这种方式处理,整个脚本的代码结构会变得十分的丑陋,而且,如果我们忘记了检查命令返回值,可能会导致脚本程序执行紊乱,造成系统不安全。 -e选项就是解决这个问题的一种十分优雅的方案,-e表示如果一个命令以非零状态退出,则整个shell脚本程序就会立即退出。比如,


#!/usr/bin/env bash
PID=`pidof dockers`
echo "pidof return 1"
如果,我们没有添加set -e,该脚本的执行结果为:
pidof return 1


加上set -e之后,脚本的执行结果为:


在这里插入代码片


可以看到,shell在检测到pidof返回1之后,就会立即退出,不会继续执行,从而可以保证脚本安全的退出。 可是有的时候,命令返回1并不代表执行失败,这时如果启用了set -e,那么脚本就会立即退出,这不是我们想看到的,解决办法有两种:


#!/usr/bin/env bash
set -e 
... ...
set +e 
command 1
command 2
set -e 
... ... 


通过 set +e关闭 -e选项,通过set -e再次打卡-e选项。


#! /usr/bin/env bash
set -e
... ...
command1 || true
退出


-o errexit等同于-e选项。


变量未定义


shell脚本中,如果遇到未定义的变量,一般会按照空值来处理,比如,


#!/usr/bin/env bash
echo $NONE
echo "NONE not set"


NONE变量从未定义过,打印NONE时会显示为空,而后继续打印下面的语句。这是不安全的,我们希望在遇到未定义的变量时,shell应该提示,并立即退出,-u选项可以完美的解决这个问题。在上面的脚本中加上set -u,再次执行,结果如下:


#!/usr/bin/env bash
set -u
echo $NONE
echo "NONE not set"


结果如下:


$ /set.sh: 行 18: NONE:未绑定的变量


可以看到,shell在检测到未定义变量NONE之后,立即提示并退出了shell。-o nounset等同于-u选项。


跟踪调试


在编写较大规模的shell脚本时,不可能一次就编写出正确的脚本,当出现bug时,我们可能希望可以看一下脚本的执行流程,-x选项可以打印当前执行的命令,方便调试、跟踪脚本的执行流程。


#! /usr/bin/env bash
set -x
ls set.sh
echo "ABC"
+ ls set.sh
set.sh
+ echo ABC
ABC


+号后面表示当前执行的命令,-o xtrace等同于-x选项。


管道


管道作为shell处理的终极武器,被人们经常用来处理复杂的业务处理,可以在一个管道命令序列中,只要最后的命令执行成功,整体的管道命令序列处理就会返回0,这使得我们不能跟踪管道中间命令的执行结果,这时-e选项就失去了作用,比如,


#! /usr/bin/env bash
set -e
cmd | echo "abc"
echo "efg"
执行结果如下:
abc
./set.sh: 行 23: cmd:未找到命令
efg


可以看到,虽然cmd命令执行失败,但是"efg"依然打印了出来。所以,对于管道命令,set -e失去了约束,shell 提供-o pipefail选项来检测管道命令的执行状态,开启这个选项之后,只要管道中存在失败的命令,shell脚本就会立即退出。


#! /usr/bin/env bash
set -eo pipefail
cmd | echo "abc"
echo "efg"
执行结果:
abc
./set.sh: 行 23: cmd:未找到命令


总结


经过上面的分析,相信大家已经了解了关于set的几个常用的选项以及它们的具体含义,它们是如此的重要,以至于在任何脚本的开头都要将它们添加上去,这样不但会提供脚本的编写、调试效率还能避免很多安全陷阱。一般的书写方式如下:


#! /usr/bin/env bash
set -eux
set -o pipefail



相关文章
|
1月前
|
Linux 网络安全 数据安全/隐私保护
Linux 超级强大的十六进制 dump 工具:XXD 命令,我教你应该如何使用!
在 Linux 系统中,xxd 命令是一个强大的十六进制 dump 工具,可以将文件或数据以十六进制和 ASCII 字符形式显示,帮助用户深入了解和分析数据。本文详细介绍了 xxd 命令的基本用法、高级功能及实际应用案例,包括查看文件内容、指定输出格式、写入文件、数据比较、数据提取、数据转换和数据加密解密等。通过掌握这些技巧,用户可以更高效地处理各种数据问题。
96 8
|
1月前
|
监控 Linux
如何检查 Linux 内存使用量是否耗尽?这 5 个命令堪称绝了!
本文介绍了在Linux系统中检查内存使用情况的5个常用命令:`free`、`top`、`vmstat`、`pidstat` 和 `/proc/meminfo` 文件,帮助用户准确监控内存状态,确保系统稳定运行。
276 6
|
1月前
|
Linux
在 Linux 系统中,“cd”命令用于切换当前工作目录
在 Linux 系统中,“cd”命令用于切换当前工作目录。本文详细介绍了“cd”命令的基本用法和常见技巧,包括使用“.”、“..”、“~”、绝对路径和相对路径,以及快速切换到上一次工作目录等。此外,还探讨了高级技巧,如使用通配符、结合其他命令、在脚本中使用,以及实际应用案例,帮助读者提高工作效率。
80 3
|
1月前
|
监控 安全 Linux
在 Linux 系统中,网络管理是重要任务。本文介绍了常用的网络命令及其适用场景
在 Linux 系统中,网络管理是重要任务。本文介绍了常用的网络命令及其适用场景,包括 ping(测试连通性)、traceroute(跟踪路由路径)、netstat(显示网络连接信息)、nmap(网络扫描)、ifconfig 和 ip(网络接口配置)。掌握这些命令有助于高效诊断和解决网络问题,保障网络稳定运行。
74 2
|
16天前
|
Linux Shell
Linux 10 个“who”命令示例
Linux 10 个“who”命令示例
45 14
Linux 10 个“who”命令示例
|
5天前
|
Ubuntu Linux
Linux 各发行版安装 ping 命令指南
如何在不同 Linux 发行版(Ubuntu/Debian、CentOS/RHEL/Fedora、Arch Linux、openSUSE、Alpine Linux)上安装 `ping` 命令,详细列出各发行版的安装步骤和验证方法,帮助系统管理员和网络工程师快速排查网络问题。
61 20
|
6天前
|
网络协议 Linux 应用服务中间件
kali的常用命令汇总Linux
kali的常用命令汇总linux
26 7
|
25天前
|
Linux 数据库
Linux中第一次使用locate命令报错?????
在Linux CentOS7系统中,使用`locate`命令时出现“command not found”错误,原因是缺少`mlocate`包。解决方法是通过`yum install mlocate -y`或`apt-get install mlocate`安装该包,并执行`updatedb`更新数据库以解决后续的“can not stat”错误。
31 9
|
24天前
|
监控 网络协议 Linux
Linux netstat 命令详解
Linux netstat 命令详解
|
1月前
|
运维 监控 网络协议
运维工程师日常工作中最常用的20个Linux命令,涵盖文件操作、目录管理、权限设置、系统监控等方面
本文介绍了运维工程师日常工作中最常用的20个Linux命令,涵盖文件操作、目录管理、权限设置、系统监控等方面,旨在帮助读者提高工作效率。从基本的文件查看与编辑,到高级的网络配置与安全管理,这些命令是运维工作中的必备工具。
118 3