read 命令

    read命令的格式如下。

    上面语法中,options是参数选项,variable是用来保存输入数值的一个或多个变量名。如果没有提供变量名,环境变量REPLY会包含用户输入的一整行数据。

    下面是一个例子demo.sh

    1. #!/bin/bash
    2. echo -n "输入一些文本 > "
    3. read text
    4. echo "你的输入:$text"

    上面例子中,先显示一行提示文本,然后会等待用户输入文本。用户输入的文本,存入变量text,在下一行显示出来。

    1. $ bash demo.sh
    2. 输入一些文本 > 你好,世界
    3. 你的输入:你好,世界

    read可以接受用户输入的多个值。

    1. #!/bin/bash
    2. echo Please, enter your firstname and lastname
    3. read FN LN
    4. echo "Hi! $LN, $FN !"

    上面例子中,read根据用户的输入,同时为两个变量赋值。

    如果用户的输入项少于read命令给出的变量数目,那么额外的变量值为空。如果用户的输入项多于定义的变量,那么多余的输入项会包含到最后一个变量中。

    如果read命令之后没有定义变量名,那么环境变量REPLY会包含所有的输入。

    1. #!/bin/bash
    2. # read-single: read multiple values into default variable
    3. echo -n "Enter one or more values > "
    4. read
    5. echo "REPLY = '$REPLY'"

    上面脚本的运行结果如下。

    read命令除了读取键盘输入,可以用来读取文件。

    1. while read myline
    2. do
    3. echo "$myline"
    4. done < $filename

    上面的例子通过read命令,读取一个文件的内容。done命令后面的定向符<,将文件导向read命令,每次读取一行,存入变量myline,直到文件读取完毕。

    (1)-t 参数

    read命令的-t参数,设置了超时的秒数。如果超过了指定时间,用户仍然没有输入,脚本将放弃等待,继续向下执行。

    1. echo -n "输入一些文本 > "
    2. if read -t 3 response; then
    3. echo "用户已经输入了"
    4. else
    5. echo "用户没有输入"
    6. fi

    上面例子中,输入命令会等待3秒,如果用户超过这个时间没有输入,这个命令就会执行失败。if根据命令的返回值,转入else代码块,继续往下执行。

    环境变量TMOUT也可以起到同样作用,指定read命令等待用户输入的时间(单位为秒)。

    1. $ TMOUT=3
    2. $ read response

    上面例子也是等待3秒,如果用户还没有输入,就会超时。

    (2)-p 参数

    -p参数指定用户输入的提示信息。

    1. read -p "Enter one or more values > "
    2. echo "REPLY = '$REPLY'"

    上面例子中,先显示Enter one or more values >,再接受用户的输入。

    (3)-a 参数

    -a参数把用户的输入赋值给一个数组,从零号位置开始。

    上面例子中,用户输入被赋值给一个数组people,这个数组的2号成员就是dodo

    -n参数指定只读取若干个字符作为变量值,而不是整行读取。

    1. $ read -n 3 letter
    2. abcdefghij
    3. $ echo $letter
    4. abc

    上面例子中,变量letter只包含3个字母。

    (5)-e 参数

    -e参数允许用户输入的时候,使用readline库提供的快捷键,比如自动补全。具体的快捷键可以参阅《行操作》一章。

    1. #!/bin/bash
    2. echo Please input the path to the file:
    3. read -e fileName
    4. echo $fileName

    上面例子中,read命令接受用户输入的文件名。这时,用户可能想使用 Tab 键的文件名“自动补全”功能,但是read命令的输入默认不支持readline库的功能。-e参数就可以允许用户使用自动补全。

    (6)其他参数

    • -d delimiter:定义字符串delimiter的第一个字符作为用户输入的结束,而不是一个换行符。
    • :raw 模式,表示不把用户输入的反斜杠字符解释为转义字符。
    • -u fd:使用文件描述符fd作为输入。

    read命令读取的值,默认是以空格分隔。可以通过自定义环境变量IFS(内部字段分隔符,Internal Field Separator 的缩写),修改分隔标志。

    IFS的默认值是空格、Tab 符号、换行符号,通常取第一个(即空格)。

    如果把IFS定义成冒号(:)或分号(;),就可以分隔以这两个符号分隔的值,这对读取文件很有用。

    1. #!/bin/bash
    2. # read-ifs: read fields from a file
    3. FILE=/etc/passwd
    4. read -p "Enter a username > " user_name
    5. file_info="$(grep "^$user_name:" $FILE)"
    6. if [ -n "$file_info" ]; then
    7. IFS=":" read user pw uid gid name home shell <<< "$file_info"
    8. echo "User = '$user'"
    9. echo "UID = '$uid'"
    10. echo "GID = '$gid'"
    11. echo "Full Name = '$name'"
    12. echo "Home Dir. = '$home'"
    13. echo "Shell = '$shell'"
    14. else
    15. echo "No such user '$user_name'" >&2
    16. exit 1
    17. fi

    上面例子中,IFS设为冒号,然后用来分解/etc/passwd文件的一行。IFS的赋值命令和read命令写在一行,这样的话,IFS的改变仅对后面的命令生效,该命令执行后IFS会自动恢复原来的值。如果不写在一行,就要采用下面的写法。

    1. OLD_IFS="$IFS"
    2. IFS=":"
    3. read user pw uid gid name home shell <<< "$file_info"

    另外,上面例子中,<<<是 Here 字符串,用于将变量值转为标准输入,因为read命令只能解析标准输入。

    上面的命令可以逐行读取文件,每一行存入变量line,打印出来以后再读取下一行。