我在以下代码中遇到了 WINCH 信号问题,该代码来自位于以下位置的教程developerWorks®:
#!/bin/bash
trap 'get_window_size' WINCH # trap when a user has resized the window
_UNDERLINE_ON=`tput smul` # turn on underline
_UNDERLINE_OFF=`tput rmul` # turn off underline
get_window_size() {
_WINDOW_X=`tput lines`
_WINDOW_Y=`tput cols`
_FULL_SPACES=`echo ""|awk '
{
_SPACES = '${_WINDOW_Y}'
while (_SPACES-- > 0) printf (" ")
}'`
_FULL_UNDERLINE=`echo "${_UNDERLINE_ON}${_FULL_SPACES}${_UNDERLINE_OFF}"`
unset _FULL_SPACES
show_menu
return 0
}
set_color() {
tput clear
PS3="Enter Selection[1-9]:"
select _COLOR in "Black" "Blue" "Green" "Cyan" "Red" "Magenta" "Yellow" "White" "Exit"
do
case ${REPLY} in
[1-8]) _X=`expr ${REPLY} - 1`;;
9) break;;
*) echo "Invalid Color"; continue;;
esac
if [[ ${1} = "b" ]]
then
tput setb ${_X}
else
tput setf ${_X}
fi
done
}
show_menu() {
while [[ -z ${_ANS} ]]
do
tput civis
tput clear
cat <<- EOF
Window Size: ${_WINDOW_X} / ${_WINDOW_Y}
Select => ${_UNDERLINE_ON} ${_UNDERLINE_OFF}
${_FULL_UNDERLINE}
B) Background Text Color
F) Foreground Text Color
X) Exit
EOF
tput rc
tput smul
tput cnorm
read _ANS
tput rmul
case ${_ANS} in
[Bb]) set_color "b";;
[Ff]) set_color "f";;
[Xx]) tput clear; exit;;
*)
echo -e "Invalid Selection: ${_ANS}\c"
sleep 2
;;
esac
unset _ANS
done
}
tput sgr0
tput civis
tput clear
tput cup 3 10
tput sc
tput cup 0 0
[[ -n ${_ANS} ]] && unset _ANS
get_window_size
exit 0
既然我正在困住绞车(换窗)信号,我希望(每次调整包含此运行脚本的窗口时)将调用 get_window_size 函数。
然而,在我的机器上,它只捕获一次 WINCH 信号(在特定条件下)。仅当我使用 xterm(在本例中为 gnome-terminal)Terminal
菜单选择默认大小之一(80x24、80x43、132x24、132x43)时,才会发生该陷阱;此外,这种情况只会发生第一次。如果我随后更改窗口大小(使用Terminal
菜单项),则不会发生陷印。
另外,如果我不使用Terminal
菜单项,而是使用指点设备重新调整大小,则即使第一个陷阱也不会发生。
有人可以帮助我理解出了什么问题(无论是代码还是我的期望)?
答案1
我怀疑脚本等待用户操作和/或信号的方式有些愚蠢。请注意,get_window_size
调用show_menu
,它等待用户输入 ( read _ANS
)。
该trap
命令还会导致get_window_size
由 SIGWINCH 调用。我不知道 shell 函数是否可重入。
我对如何生成 SIGWINCH 以及何时执行 trap 函数的期望与您的基本相同。我稍微删减了示例脚本,并使其不可重入。见下文。
您可能还需要考虑使用不同的窗口管理器尝试您的脚本。我twm
在实验中使用了古老的、快速的和高度可配置的。
#!/bin/bash
trap 'get_window_size' WINCH # trap when a user has resized the window
get_window_size() {
_WINDOW_X=`tput lines`
_WINDOW_Y=`tput cols`
echo "X: $_WINDOW_X"
echo "Y: $_WINDOW_Y"
return 0
}
while read ALINE
do
echo "Read: '$ALINE'"
done