我有一个bash文件,它提示用户输入一些参数,如果没有给出任何参数则使用默认值。然后该脚本继续使用参数执行一些其他命令。 这很有效 - 直到最近添加才有问题。
在尝试从txt文件中读取NAMES参数时,我添加了一个while循环来接收文件中的名称,但我仍然希望提示剩下的参数。
但是一旦我添加了while循环,输出就会在get_ans()中显示打印的提示,并且永远不会暂停读取,因此会选择所有默认值。
我想从文件中读取第一个参数,然后从提示用户读取所有后续文件。
通过添加while循环我打破了什么?
cat list.txt |
while read line
do
get_ans "Name" "$line"
read NAME < $tmp_file
get_ans "Name" "$line"
read NAME < $tmp_file
done
function get_ans
{
if [ -f $tmp_file ]; then
rm $tmp_file
PROMPT=$1
DEFAULT=$2
echo -n "$PROMPT [$DEFAULT]: "
read ans
if [ -z "$ans" ]; then
ans="$DEFAULT"
fi
echo "$ans" > $tmp_file
}
(注意:代码不是复制和粘贴,所以请原谅错别字。实际代码在main()之前定义了函数)
答案 0 :(得分:3)
您将数据传输到while
循环STDIN。因此read
中的get_ans
也会从该STDIN流中获取数据。
您可以在不同的文件描述符上管理数据,以避免出现问题并停止使用临时文件:
while read -u 9 line; do
NAME=$(get_ans Name "$line")
done 9< list.txt
get_ans() {
local PROMPT=$1 DEFAULT=$2 ans
read -p "$PROMPT [$DEFAULT]: " ans
echo "${ans:-$DEFAULT}"
}
答案 1 :(得分:3)
直接从终端读取,而不是从stdin读取(假设您使用的是* NIX机器,而不是Windows机器):
while read foo</some/file; do
read bar</dev/tty
echo "got <$bar>"
done
答案 2 :(得分:1)
在命令行中将一个命令传递给另一个命令时,如:
$ foo | bar
shell将对其进行设置,以便bar
的标准输入来自foo
的标准输出。 foo
发送到stdout的任何内容都将直接转到bar
的标准输入。
在您的情况下,这意味着您的脚本可以读取的唯一内容是cat
命令的标准输出,它将包含文件的内容。
不要在命令行上使用管道,而是将文件名作为脚本的第一个参数。然后打开并从代码中的文件中读取并正常读取用户。