POSIX shell学习笔记

阅读量: searchstar 2024-07-29 13:39:27
Categories: Tags:

学习POSIX shell建议使用dash,因为它很快:https://unix.stackexchange.com/a/148098

man dash: Only features designated by POSIX, plus a few Berkeley extensions, are being incorporated into this shell.

条件判断

man dash,然后搜索test expression,可以看到完整的列表。

if elif else

#!/usr/bin/env sh
if [ condition1 ]; then
	# Do something
elif [ condition2 ]; then
	# Do something
else
	# Do something
fi

判断某环境变量是否存在

参考:https://blog.csdn.net/blade2001/article/details/7243143?utm_source=blogxgwz3

上面的文章好像写反了

例子:判断环境变量DISPLAY是否存在(若不存在说明没有提供显示设备)

if [ "$DISPLAY" ]; then
	# DISPLAY存在
else
	# DISPLAY不存在
fi

或者

if [ ! "$DISPLAY" ]; then
	# DISPLAY不存在

字符串

功能 例子
为空 [ -z "$1" ] 或者 [ ! "$1" ]
非空 [ -n "$1" ] 或者 [ "$1" ]
相等 [ "$1" = "$2" ]
不相等 [ "$1" != "$2" ]
字典序小于 [ "$1" \< "$2" ]
字典序大于 [ "$1" \> "$2" ]

所以如果判断目录是否非空,可以这样:if [ "$(ls -A xxx)" ]

参考:https://www.cyberciti.biz/faq/linux-unix-shell-check-if-directory-empty/

整数的大小判断

代码 含义 例子
-eq = [ $1 -eq 2 ]
-ne != [ $1 -ne 2 ]
-le <= [ $1 -le 2 ]
-lt < [ $1 -lt 2 ]
-ge >= [ $1 -ge 2 ]
-gt > [ $1 -gt 2 ]
-a && [ $1 -gt 0 -a $1 -lt 10 ]
-o || [ $1 -lt 0 -o $1 -gt 10 ]

浮点数的大小判断

if awk "BEGIN {exit !(1.234 >= .233)}"; then
    echo "yes"
fi

来源:https://stackoverflow.com/a/45591665/13688160

判断文件类型

来源:https://jingyan.baidu.com/article/95c9d20d5ac536ec4e7561ad.html

#!/usr/bin/env sh
if [ -z "$1" ]; then      #如果没有输入参数,也就是第一个参数的字符串长度为0
    :                          #空语句
else
	if [ -e "$1" ]; then       #如果文件存在的话
		if [ -f "$1" ]; then   #如果文件是个普通文件?
			echo "$1 is a text file."
		elif [ -d "$1" ]; then #如果文件是个目录文件?
			echo "$1 is a directory."
		elif [ -c "$1" ]; then #如果文件是个字符设备?
			echo "$1 is a char device."
		elif [ -b "$1" ]; then #如果文件是个块设备?
			echo "$1 is a block device."
		else #否则
			echo "$1 is unknow file."
	fi
fi

另外,-s表示文件存在且不为空。来源:https://stackoverflow.com/questions/9964823/how-to-check-if-a-file-is-empty-in-bash

判断文件权限

代码 含义
-r 存在且可读
-w 存在且可写
-x 存在且可执行

参考:https://stackoverflow.com/questions/10319652/check-if-a-file-is-executable

函数

参考:https://www.runoob.com/linux/linux-shell-func.html

定义

FunctionName() {
	do_some_thing_here
	return Interger
}

也可以不return。
参数用法与脚本类似。$#表个数,$1, $9, ${10}表具体参数。

使用

FunctionName par1 par2 par3

循环

while

参考:https://wiki.jikexueyuan.com/project/shell-tutorial/shell-while-loop.html

while Command
do
   Statement(s) to be executed if Command is true
done

或者

while [ Condition ]; do
	Statement(s) to be executed if Condition is true
done

也可以对命令返回值取反,比如

while ! Command; do
	Statement(s) to be executed if Command is false
done

重定向

# 将`stdout`重定向到`stdout.txt`
Command > stdout.txt
# 将`stderr`重定向到`stderr.txt`
Command 2> stderr.txt
# 将stderr重定向到stdout
Command 2>&1
# 将stderr和stdout都重定向到一个文件
# 参考:<https://blog.csdn.net/u011630575/article/details/52151995>
Command > shell.log 2>&1
# 将`stdin`重定向到`stdin.txt`
Command < stdin.txt
# 将Command1的stdout输入到Command2的stdin
Command1 | Command2 # 例如 echo 'whoami' | sh,可以让sh执行whoami。

将多行字面量作为stdin

Command <<标记
第一行
第二行
...
标记

例如:

sh <<EOF
whoami
echo 2333
EOF

命令行参数

arg_num() {
	# 参数个数(不含$0)
	echo $#
}
# 访问某参数
echo "$2"
# 相当于arg_num "$1" "$2"
arg_num "$@"
# 相当于arg_num "$1 $2"
arg_num "$*"
shift [n]
	Shift  the  positional  parameters  n  times.  A shift sets the value of $1 to the value of $2, the value of $2 to the
	value of $3, and so on, decreasing the value of $# by one.  If n is greater than the number of positional  parameters,
	shift will issue an error message, and exit with return status 2.

n似乎默认是1。

Command substitution

把命令的输出会保存到变量:

a=$(command args...)

但如果是要把输出当字符串用,需要在周围加上双引号。但是要注意,$()之间的内容不需要再加一层转义。举个例子,判断当前目录是不是a b

check() {
	# 而不是 "$(basename \"$(pwd)\")"
	if [ "$(basename "$(pwd)")" = "a b" ]; then
		echo yes
	else
		echo no
	fi
}
check
mkdir -p "a b"
cd "a b"
check

其中"$(basename "$(pwd)")"相当于先执行basename "$(pwd)",再把输出用双引号包起来变成字符串。

如果改成"$(basename \"$(pwd)\")",就变成执行basename \"$(pwd)\",也就是先执行pwd,然后执行basename \"a b\",结果就变成a了。

相关文档:https://pubs.opengroup.org/onlinepubs/9699919799/utilities/V3_chap02.html

进程

获取当前subshell的PID:

pid=$(exec sh -c 'echo "$PPID"')

来源:https://unix.stackexchange.com/a/484464

set

可以用set来设置一些模式。

子命令返回值不为0时退出

set -e

但是要注意的是,对于用管道连接起来的几个命令,最终的exit code似乎是管道里的最后一个命令。例如对于command A | command B,如果command A出错返回非0,但是command B正常退出,那么这个组合命令的exit code会被设置成0,这样set -e就不会生效。

bash支持set -o pipefail,其效果是将exit code赋值为被管道组合起来的命令中最后一个返回非0值的命令,这样被管道组合起来命令中任何一个命令返回非0都会退出。

来源:Get exit status of process that's piped to another

trap

格式:

trap 命令 事件

常用事件:

常用信号:
- HUP: SIGHUP,父进程退出的时候会向子进程发送SIGHUP。但注意,如果父进程是被SIGKILL杀死的,那SIGHUP不一定会发送到子进程。
- INT: SIGINT,可以由ctrl+c触发
- TERM: SIGTERM

例子:

# a.sh
trap 'exit 1' INT
trap 'echo cleanup' EXIT
sleep 3

sh a.sh,然后ctrl+cINT事件被触发,从而调用exit 1,进而触发EXIT事件,从而调用echo cleanup,然后屏幕上会打印一行cleanup

来源:https://newbe.dev/exit-trap-in-dash-vs-ksh-and-bash

但是要注意的是,如果脚本正在执行一个外部命令,比如sleep,这时脚本里的trap是不会被调用的。因此在需要进行信号处理的脚本中如果需要执行时间很长的命令,那么要把这个命令放到子进程里跑,然后在主进程里wait。例如test.sh

#!/usr/bin/env sh
trap 'echo exit 1; exit 1' TERM
sleep 1000000
./test.sh &
pid=$!
sleep 1
kill -TERM $pid

这时候这个进程是杀不死的,因为sleep 1000000阻塞了脚本,所以trap没有执行。但我们可以把sleep放到子进程里:

#!/usr/bin/env sh
sleep 100000 &
pid=$!
trap "kill -TERM $pid; echo exit 1; exit 1" TERM
wait

再执行上面的命令,这个脚本就可以被顺利杀死了。

如果是pipeline命令,直接将其放入后台的话,$!只是pipeline里的最后进程,将其杀掉只会导致pipeline前面的进程变成孤儿进程。为了一次性将整个pipeline杀死,可以用setsid将其放入一个单独的进程组里,然后向整个进程组发信号:

#!/usr/bin/env sh
setsid sh -c "sleep 10 | sleep 10" &
trap "kill -TERM -$!; echo exit 1; exit 1" TERM
wait

详情请参考:Linux shell向进程组发信号

wait

如果不带参数,就是等待所有后台任务完成。

也可以等待一个特定进程:wait <PID>

bash中可以wait -n来等待任意一个后台任务完成,但POSIX标准中没有这个。

在POSIX shell中如果要等待多个任务中的任意一个完成,然后把其他的杀掉,可以这样:

#!/usr/bin/env sh
# "wait -n" is not available in POSIX
trap "exit 1" CHLD
trap "pkill -P $$; exit 1" EXIT
sleep 10 &
sleep 3 &
wait

其中$$是当前shell的PID,pkill -P $$表示把父进程是当前shell的所有进程杀掉。

来源:https://unix.stackexchange.com/a/231678

如果是带pipeline的复杂命令,需要这样:

#!/usr/bin/env sh
# "wait -n" is not available in POSIX
trap "exit 1" CHLD
trap "pkill -P $$; exit 1" EXIT
(
    setsid sh -c "sleep 10 | sleep 10" &
    trap "kill -TERM -$!" TERM
    wait
) &
sleep 3 &
wait

但是需要注意的是,$$在subshell中会被展开为shell的PID,而不是subshell的PID。因此如果在subshell中,需要这样:

#!/usr/bin/env sh
(
	# https://unix.stackexchange.com/a/484464
	pid=$(exec sh -c 'echo "$PPID"')
	# "wait -n" is not available in POSIX
	trap "exit 1" CHLD
	trap "pkill -P $pid; exit 1" EXIT
	(
		setsid sh -c "sleep 10 | sleep 10" &
		trap "kill -TERM -$!" TERM
		wait
	) &
	sleep 3 &
	wait
)

case

case 表达式 in
	正则1)
		命令...
		;;
	正则2)
		命令...
		;;
	正则3|正则4)
		命令...
		;;
	*)
		默认命令...
		;;
esac

Conditional substitution

${VAR:+内容}

如果VAR存在,它就会展开成内容,如果VAR不存在,就展开成空的。

跟case配合可以实现往PATH里添加路径:

append_path () {
    case ":$PATH:" in
        *:"$1":*)
            ;;
        *)
            PATH="${PATH:+$PATH:}$1"
    esac
}

如果PATH不存在,${PATH:+$PATH:}$1就展开为$1,如果PATH存在,就展开为$PATH:$1

unset

文档:https://pubs.opengroup.org/onlinepubs/009696699/utilities/unset.html

默认是unset环境变量。但如果是要unset函数,要用unset -f

存在的问题