Shell 获得用户输入

在 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

使用效果:

Shell 获得用户输入_第1张图片
显示效果

实例:输入超时

我们可以使用 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

使用效果:

Shell 获得用户输入_第2张图片
显示效果

实例:隐藏方式读取

我们可以使用 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"

使用效果:

Shell 获得用户输入_第3张图片
显示效果

实例:从文件中读取

使用 read 命令从文件中读取数据的方法主要有两种,一种是在 while 循环或 until 循环中使用 read 命令,通过文件描述符一行一行地读取文件的内容;另一种就是本节我们要介绍的方法 —— 在 for 循环中使用命令 “cat ” 来读取文件中的内容。
在 for 循环中使用命令 “cat ” 来读取文件的语法是:

for data in $(cat filename)
do
    command1
    command2
    ...
    commandN
done

上述语法中 filename 代表一个文本文件,读取的内容会被存入变量 data,此变量可以在 for 循环体中使用,用于对读入的数据进行处理 。

使用上述方法需要注意:它是逐个单词地读取文件内容。因为使用这种方法读取文件内容时,它使用环境变量 IFS 的值作为分隔符,由于 $IFS 的默认值是 “”,所以它会首先以空格作为分隔符,来读取文件的内容。因此,如果我们想要使用这种方法逐行地读取文件的内容,就需要在调用 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

使用效果:

Shell 获得用户输入_第4张图片
显示效果

尽管使用 while 循环读取文件的内容相对比较方便,但是它也有副作用,它读取的每行内容会去掉重复的空格和制表符,即会消除每行的原有格式。而将 for 循环结合环境变量 $IFS 使用可以保留每行原有的格式。所有,我们可以根据不同的需求来选择使用 while 还是 for 循环来读取文件的内容。

本文参考自 《Linux Shell命令行及脚本编程实例详解 》

你可能感兴趣的:(Shell 获得用户输入)