在 Bash 下我们可以通过其内部命令 read 接收用户来自键盘的输入,并可以将输入的内容赋值给一个变量。
实例:基本的读取
read 命令比较常用的语法格式如下所示:
read [-p prompt] [variable1 variable2...]
-p 选项用于在尝试读取任何输入之前,显示 prompt(提示信息)的内容到标准错误输出。我们一般使用这一选项来指定提示用户输入哪些内容的信息。read 命令会每次从标准输入(或使用 -u 选项指定的文件描述符中)读取一行的内容,它会将第一个单词赋值给第一个变量 variable1,第二个单词赋值给第二个变量 variable2,依次类推。如果输入的单词数少于指定的变量数,那么剩下的 name 变量的值会被设为空,环境变量 IFS 中的字符被作为分隔符来将输入的内容分隔为单词。
下面是一个简单的使用示例:
#! /bin/bash
read -p "Enter your name, please: " username
read -p "Enter your email, please: " email
read -p "Are you sure to continue? [y/n] " input
case $input in
[yY]*)
echo "Your name is $username"
echo "Your email is $email"
;;
[nN]*)
exit
;;
*)
echo "Just enter y or n, please."
exit
;;
esac
使用效果:
实例:输入超时
我们可以使用 read 命令的 -t 选项来设置 read 命令读取用户输入时的超时时间。如果在指定的秒数内没有读入一整行的输入(即没有输入回车键),read 命令就会超时并返回失败。我们可以在上一小节的实例脚本脚本中加入读取输入超时功能,修改之后的脚本的内容如下:
#! /bin/bash
read -t 5 -p "Enter your name, please: " username
read -t 5 -p "Enter your email, please: " email
read -t 5 -p "Are you sure to continue? [y/n] " input
case $input in
[yY]*)
echo "Your name is $username"
echo "Your email is $email"
;;
[nN]*)
exit
;;
*)
echo "Just enter y or n, please."
exit
;;
esac
使用效果:
实例:隐藏方式读取
我们可以使用 read 命令的 -s 选项来隐藏用户的输入。如果指定了 -s 选项,则来自终端的输入不会被显示出来。这对我们的脚本在需要对用户输入的密码进行验证时(为了安全)是很有用的。
下面是一个简单的使用示例:
#! /bin/bash
password=''
echo -n "Enter password: "
# 使用 while 循环隐式地从标准输入每次读取一个字符,且反斜杠不做转义字符处理
# 然后将读取的字符赋值给变量 char
while IFS= read -r -s -n1 char
do
# 如果读入的字符为空,则退出 while 循环
if [ -z $char ]
then
echo
break
fi
# 如果输入的是退格或删除键,则移除一个字符
if [[ $char == $'\x08' || $char == $'\x7f' ]]
then
[[ -n $password ]] && password=${password:0:${#password}-1}
printf '\b \b'
else
password+=$char
printf '*'
fi
done
echo "Password is: $password"
使用效果:
实例:从文件中读取
使用 read 命令从文件中读取数据的方法主要有两种,一种是在 while 循环或 until 循环中使用 read 命令,通过文件描述符一行一行地读取文件的内容;另一种就是本节我们要介绍的方法 —— 在 for 循环中使用命令 “cat <filename>” 来读取文件中的内容。
在 for 循环中使用命令 “cat <filename>” 来读取文件的语法是:
for data in $(cat filename)
do
command1
command2
...
commandN
done
上述语法中 filename 代表一个文本文件,读取的内容会被存入变量 data,此变量可以在 for 循环体中使用,用于对读入的数据进行处理 。
使用上述方法需要注意:它是逐个单词地读取文件内容。因为使用这种方法读取文件内容时,它使用环境变量 IFS 的值作为分隔符,由于 $IFS 的默认值是 “<space><tab><newline>”,所以它会首先以空格作为分隔符,来读取文件的内容。因此,如果我们想要使用这种方法逐行地读取文件的内容,就需要在调用 for 循环之前先修改 $IFS 的值。当然,如果我们能确定文件的内是每行只有一个单词,就无需进行修改。
#! /bin/bash
old_IFS=$IFS
# 参数个数不为 1
if [ $# -ne 1 ]
then
echo "Usage: `basename $0` filename"
exit
fi
# 如果指定的文件不存在
if [ ! -f $1 ]
echo "The file $1 doesn't exist!"
exit 1
fi
# 修改环境变量 IFS 的值,使下面的 for 循环以换行符作为分隔符
IFS=$'\n'
for line in $(cat $1)
do
echo $line
done
IFS=$old_IFS
使用效果:
尽管使用 while 循环读取文件的内容相对比较方便,但是它也有副作用,它读取的每行内容会去掉重复的空格和制表符,即会消除每行的原有格式。而将 for 循环结合环境变量 $IFS 使用可以保留每行原有的格式。所有,我们可以根据不同的需求来选择使用 while 还是 for 循环来读取文件的内容。
本文参考自 《Linux Shell命令行及脚本编程实例详解 》