Linux Bash Set命令解析

简介: 平时在学习大牛的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



相关文章
|
Unix Linux
对于Linux的进程概念以及进程状态的理解和解析
现在,我们已经了解了Linux进程的基础知识和进程状态的理解了。这就像我们理解了城市中行人的行走和行为模式!希望这个形象的例子能帮助我们更好地理解这个重要的概念,并在实际应用中发挥作用。
252 20
|
Ubuntu Linux
"unzip"命令解析:Linux下如何处理压缩文件。
总的来说,`unzip`命令是Linux系统下一款实用而方便的ZIP格式文件处理工具。本文通过简明扼要的方式,详细介绍了在各类Linux发行版上安装 `unzip`的方法,以及如何使用 `unzip`命令进行解压、查看和测试ZIP文件。希望本文章能为用户带来实际帮助,提高日常操作的效率。
3304 12
|
Linux
Linux命令的基本格式解析
总的来说,Linux命令的基本格式就像一个食谱,它可以指导你如何使用你的计算机。通过学习和实践,你可以成为一个真正的“计算机厨师”,创造出各种“美味”的命令。
364 15
|
存储 Linux
Linux内核中的current机制解析
总的来说,current机制是Linux内核中进程管理的基础,它通过获取当前进程的task_struct结构的地址,可以方便地获取和修改进程的信息。这个机制在内核中的使用非常广泛,对于理解Linux内核的工作原理有着重要的意义。
633 11
|
监控 Shell Linux
Android调试终极指南:ADB安装+多设备连接+ANR日志抓取全流程解析,覆盖环境变量配置/多设备调试/ANR日志分析全流程,附Win/Mac/Linux三平台解决方案
ADB(Android Debug Bridge)是安卓开发中的重要工具,用于连接电脑与安卓设备,实现文件传输、应用管理、日志抓取等功能。本文介绍了 ADB 的基本概念、安装配置及常用命令。包括:1) 基本命令如 `adb version` 和 `adb devices`;2) 权限操作如 `adb root` 和 `adb shell`;3) APK 操作如安装、卸载应用;4) 文件传输如 `adb push` 和 `adb pull`;5) 日志记录如 `adb logcat`;6) 系统信息获取如屏幕截图和录屏。通过这些功能,用户可高效调试和管理安卓设备。
10241 2
|
运维 Shell 数据库
Python执行Shell命令并获取结果:深入解析与实战
通过以上内容,开发者可以在实际项目中灵活应用Python执行Shell命令,实现各种自动化任务,提高开发和运维效率。
525 20
|
网络协议 Unix Linux
深入解析:Linux网络配置工具ifconfig与ip命令的全面对比
虽然 `ifconfig`作为一个经典的网络配置工具,简单易用,但其功能已经不能满足现代网络配置的需求。相比之下,`ip`命令不仅功能全面,而且提供了一致且简洁的语法,适用于各种网络配置场景。因此,在实际使用中,推荐逐步过渡到 `ip`命令,以更好地适应现代网络管理需求。
853 11
|
存储 运维 安全
深入解析操作系统控制台:阿里云Alibaba Cloud Linux(Alinux)的运维利器
本文将详细介绍阿里云的Alibaba Cloud Linux操作系统控制台的功能和优势。
545 6
|
监控 安全 Shell
防止员工泄密的措施:在Linux环境下使用Bash脚本实现日志监控
在Linux环境下,为防止员工泄密,本文提出使用Bash脚本进行日志监控。脚本会定期检查系统日志文件,搜索敏感关键词(如"password"、"confidential"、"secret"),并将匹配项记录到临时日志文件。当检测到可疑活动时,脚本通过curl自动将数据POST到公司内部网站进行分析处理,增强信息安全防护。
535 0
|
Linux Shell Windows
4:Bash shell命令-步入Linux的现代方法
4:Bash shell命令-步入Linux的现代方法
324 0