Linux Bash Set命令解析

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 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



相关文章
|
23小时前
|
关系型数据库 MySQL Linux
Linux命令systemctl详解
`systemctl`是Linux系统用于管理systemd服务的核心命令,它与systemd守护进程交互,实现启动、停止、重启服务及查看服务状态等功能。主要参数包括`start`、`stop`、`restart`、`status`、`enable`和`disable`等。例如,启动Apache服务使用`systemctl start httpd.service`,查看服务状态用`systemctl status <service>`。使用时需注意权限,服务名通常以`.service`结尾,但命令中可省略。最佳实践包括利用tab键补全、定期查看服务状态和合理配置服务自启。
|
23小时前
|
存储 安全 Linux
Linux命令sync详解
`sync`命令在Linux中用于将内存缓冲区的数据强制写入磁盘,保证数据持久性和一致性。它在关机、重启或重要文件操作前后使用,以防数据丢失。工作原理是强制将内存中的数据同步到磁盘,特点是阻塞式执行且通常无需参数。常见用法包括安全关机、数据备份和配置文件修改后确保更改生效。应注意,过度使用可能影响性能,应适时使用`fsck`检查文件系统一致性。
|
23小时前
|
存储 算法 安全
Linux命令sum详解
`sum`命令在Linux中用于计算文件的校验和与磁盘块数,确保文件传输或存储时的完整性。它使用加法运算生成校验和,可与文件内容比较验证变化。支持不同算法(如CRC),能处理多个文件。基本用法包括:`sum file.txt`来计算校验和,`sum -c checksum.txt`来验证文件完整性。但要注意,更强的校验算法如MD5或SHA家族可能更适合安全性需求。结合`find`和`xargs`可用于目录的递归校验。定期校验和记录校验和是最佳实践。
|
23小时前
|
安全 数据管理 Shell
Linux命令su详解
`su`命令在Linux中用于切换用户身份,常用于权限管理。它允许用户无须注销当前会话就切换到另一个用户,尤其是root。`su`有多种选项,如`-`或`--login`加载目标用户环境,`-c`执行指定命令后返回。使用时需注意权限安全,建议用`sudo`以减少风险。通过限制`/etc/pam.d/su`可加强访问控制。`su`在系统维护和数据管理中扮演角色,但不直接处理数据。
|
23小时前
|
Unix Linux 数据处理
Linux命令stty详解
`stty`是Linux命令,用于设置和查看终端参数,如波特率、字符处理和控制字符。它直接与终端驱动交互,支持多种选项以适应不同的配置需求。例如,`stty -a`显示当前设置,`stty -echo`关闭回显,`stty 115200 cs8`调整波特率和字符大小。注意修改设置可能影响终端行为,建议先备份(`stty -g`)并谨慎操作。查阅手册页以获取详细信息。
|
23小时前
|
安全 Linux 数据处理
Linux命令strip详解
`strip`命令在Linux中用于移除可执行文件和库的符号表及调试信息,减小文件大小,提升运行效率。它的工作原理是删除文件中包含的函数名、变量名等信息。主要参数包括`-s`(移除所有符号)、`-g`(仅移除调试信息)等。在应用时要注意文件备份,因为该操作不可逆。最佳实践是在发布版本中使用,并结合构建流程自动化。
|
23小时前
|
安全 Linux 数据安全/隐私保护
Linux命令strings详解
`strings`是Linux工具,用于从二进制文件中提取可打印字符串,常用于文件分析、安全审计和逆向工程。它可以识别至少4个连续可打印字符的序列,并支持多种参数,如`-n`调整最小长度,`-f`显示文件名。示例用法包括`strings /bin/ls`和`strings -n 6 /usr/bin/uptime | grep GLIBC`。注意敏感信息泄露,结合其他命令可增强分析能力。
|
23小时前
|
存储 监控 Linux
stdbuf命令在Linux中的深度解析
`stdbuf`是Linux工具,用于控制命令的stdin、stdout和stderr的缓冲模式。它可以设置为无缓冲、行缓冲或块缓冲,以优化数据处理和实时性。例如,`stdbuf -o0 cmd`禁用cmd的输出缓冲,`-oL`则按行缓冲。在需要实时监控或高效处理大量数据时,选择合适的缓冲模式至关重要。注意,过度使用无缓冲可能影响性能,并非所有系统都支持`stdbuf`。
|
23小时前
|
存储 运维 安全
Linux命令stat:深入了解文件与文件系统状态
`stat`命令在Linux中用于显示文件和文件系统的详细状态,包括权限、大小、时间戳等。它通过读取inode获取信息,特点是显示全面、易用且支持多种参数,如`-c`自定义格式,`-f`查看文件系统状态,`-L`处理符号链接。例如,`stat example.txt`显示文件详情,`stat -c "%n 的大小是 %s 字节" example.txt`输出文件大小。理解`stat`有助于系统管理和故障排查。
|
23小时前
|
监控 安全 Linux
Linux命令ssltap的深入解析与应用实践
`ssltap`是一个假想的Linux命令,用于模拟SSL/TLS流量分析。它捕获、解密(如果有密钥)并分析加密流量,提供实时监控、协议解析和安全审计。特点包括实时性、灵活性、可扩展性和安全性。示例用法包括捕获特定端口流量和实时监控会话状态。在实际操作中应注意私钥安全、性能影响及合规性,建议定期审计和自动化监控。