在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 命令的-弋选项来设置 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 lFS= 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 'lb \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 $O` 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循环来读取文件的内容。
本文来自:https://wenku.baidu.com/view/d13d32af87868762caaedd3383c4bb4cf7ecb73d.html
经过了手工校对,修正了几个小错误。
本站文章除注明转载/出处外,均为原创,若要转载请务必注明出处。转载后请将转载链接通过邮件告知我站,谢谢合作。本站邮箱:admin@only4.work
尊重他人劳动成果,共创和谐网络环境。点击版权声明查看本站相关条款。