我想读取一个文件并将其保存在变量中,但我需要保留变量而不仅仅是打印出文件。 我该怎么做呢?我已经写了这个脚本,但它不是我所需要的:

#!/bin/sh
while read LINE  
do  
  echo $LINE  
done <$1  
echo 11111-----------  
echo $LINE  

在我的脚本中,我可以将文件名作为参数,因此,如果文件包含“aaaa”,例如,它将打印出以下内容:

aaaa
11111-----

但这只是将文件打印到屏幕上,我想将它保存到一个变量中! 有什么简单的方法吗?


当前回答

这对我来说很管用: v = $ (cat < file_path >) echo $ v

其他回答

如果你想把整个文件读入一个变量:

#!/bin/bash
value=`cat sources.xml`
echo $value

如果你想逐行阅读:

while read line; do    
    echo $line    
done < file.txt

在跨平台中,使用最低公分母sh:

#!/bin/sh
value=`cat config.txt`
echo "$value"

在bash或zsh中,在不调用cat的情况下将整个文件读入变量:

#!/bin/bash
value=$(<config.txt)
echo "$value"

在bash或zsh中调用cat来获取文件将被认为是无用的使用cat。

注意,没有必要为命令替换加上引号来保留换行符。

参见:Bash黑客的Wiki -命令替换-专长。

两个重要的陷阱

到目前为止,其他答案都忽略了这一点:

从命令展开中删除尾随换行符 NUL字符删除

从命令展开中删除尾随换行符

这是一个问题:

value="$(cat config.txt)"

类型解决方案,但不是基于读的解决方案。

命令展开会移除尾随的换行:

S="$(printf "a\n")"
printf "$S" | od -tx1

输出:

0000000 61
0000001

这打破了从文件中读取的简单方法:

FILE="$(mktemp)"
printf "a\n\n" > "$FILE"
S="$(<"$FILE")"
printf "$S" | od -tx1
rm "$FILE"

POSIX解决方案:在命令展开部分附加一个额外的字符,然后将其删除:

S="$(cat $FILE; printf a)"
S="${S%a}"
printf "$S" | od -tx1

输出:

0000000 61 0a 0a
0000003

几乎POSIX的解决方案:ASCII编码。见下文。

NUL字符删除

在变量中存储NUL字符没有合理的Bash方法。

这会影响扩展和读取解决方案,我不知道有什么好的解决方案。

例子:

printf "a\0b" | od -tx1
S="$(printf "a\0b")"
printf "$S" | od -tx1

输出:

0000000 61 00 62
0000003

0000000 61 62
0000002

哈,我们的NUL不见了!

解决方法:

ASCII编码。见下文。 使用bash扩展$""字面值: S =美元“b \ 0” 打印“$S”| od -tx1 只适用于字面量,所以对从文件中读取没有用处。

解决陷阱的方法

在变量中存储文件的uuencode base64编码版本,并在每次使用前解码:

FILE="$(mktemp)"
printf "a\0\n" > "$FILE"
S="$(uuencode -m "$FILE" /dev/stdout)"
uudecode -o /dev/stdout <(printf "$S") | od -tx1
rm "$FILE"

输出:

0000000 61 00 0a
0000003

uuencode和udecode是POSIX 7,但在Ubuntu 12.04默认不是(sharutils包)…我没有看到POSIX 7替代bash进程<()替换扩展除了写入另一个文件…

当然,这又慢又不方便,所以我想真正的答案是:如果输入文件可能包含NUL字符,就不要使用Bash。

所有给定的解都非常慢,所以:

mapfile -d '' content </etc/passwd  # Read file into an array
content="${content[*]%$'\n'}"       # Remove trailing newline

更好地优化它,但我不能想到太多

更新:找到一个更快的方法

read -rd '' content </etc/passwd

这将返回退出代码1,所以如果你需要它 总是0:

read -rd '' content </etc/passwd || :

我使用:

NGINX_PID=`cat -s "/sdcard/server/nginx/logs/nginx.pid" 2>/dev/null`

if [ "$NGINX_PID" = "" ]; then
  echo "..."
  exit
fi