我正在编写一个交互式shell脚本,需要在尽可能多的系统上运行。是否有另一种方法可以实现与标准POSIX系统兼容的以下方法?
#! /bin/bash
echo -n "Do you wish to continue? (Y/n) 5 seconds to respond... "
read -n 1 -t 5 answer # accepts a single character, 5 second timeout.
if [ "$answer" = "n" ] || [ "$answer" = "N" ] ; then
echo -e "\nExiting..."
exit
fi
echo -e "\nContinuing with script..."
# More code
read
上的超时对我来说最重要(read -t 5
)。读取的一个字符限制是可取的,但不是必需的(read -n 1
)。
理想情况下,该脚本可以在POSIX系统上运行,也可以在bash中运行,而无需启用特殊的POSIX兼容模式。
答案 0 :(得分:4)
根据Dickey先生的回答调整stty设置,如果按下“n”或“N”以外的任何内容,以下内容似乎可以正常工作并继续使用。
据我所知,所有stty设置都是posix。
#!/bin/sh
read_char() {
old=$(stty -g)
stty raw -echo min 0 time 50
eval "$1=\$(dd bs=1 count=1 2>/dev/null)"
stty $old
}
printf "Do you wish to continue? (Y/n) 5 seconds to respond..."
read_char answer
# answer=$(getch)
# answer=$(getche)
if [ "$answer" = "n" ] || [ "$answer" = "N" ] ; then
printf "\nExiting...\n"
exit
fi
printf "\nContinuing with script...\n"
“read_char”的替代选项:
这避免了使用eval(可能不安全)
getch() {
old=$(stty -g)
stty raw -echo min 0 time 50
printf '%s' $(dd bs=1 count=1 2>/dev/null)
stty $old
}
这可以避免使用eval并打印按下的键
getche() {
old=$(stty -g)
stty raw min 0 time 50
printf '%s' $(dd bs=1 count=1 2>/dev/null)
stty $old
}
答案 1 :(得分:1)
stty
程序提供了执行此操作的方法。 xterm
有几个脚本(在其源代码的“vttests”子目录中),它保存,修改和恢复终端设置,以允许它读取终端对转义序列的响应。以下是dynamic2.sh
(开头设置printf
或echo
的一部分,用于解决一些带有$CMD
变量的旧系统:
echo "reading current color settings"
exec </dev/tty
old=`stty -g`
stty raw -echo min 0 time 5
original=
for N in $FULL
do
$CMD $OPT "${ESC}]$N;?^G${SUF}" > /dev/tty
read reply
eval original$N='${reply}${SUF}'
original=${original}${reply}${SUF}
done
stty $old
if ( trap "echo exit" EXIT 2>/dev/null ) >/dev/null
then
trap '$CMD $OPT "$original" >/dev/tty; exit' EXIT HUP INT TRAP TERM
else
trap '$CMD $OPT "$original" >/dev/tty; exit' 0 1 2 5 15
fi