小墨の博客

梦想需要付诸行动,否则只能是梦

Shell基础——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


实例:输入超时

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

尊重他人劳动成果,共创和谐网络环境。点击版权声明查看本站相关条款。

    发表评论:

    搜索
    本文二维码
    标签列表
    站点信息
    • 文章总数:508
    • 页面总数:20
    • 分类总数:92
    • 标签总数:208
    • 评论总数:61
    • 浏览总数:225323

    | | |
    | |  Z-Blog PHP