为了账号安全,请及时绑定邮箱和手机立即绑定

如何避免bash命令替换以删除换行符?

/ 猿问

如何避免bash命令替换以删除换行符?

婷婷同学_ 2019-12-27 09:36:10

为了加快执行bash脚本的速度,我想使用命令替换将命令的结果保留在变量中,但是命令替换将0x0A换行符替换为空格。例如:


a=`df -H`

要么


a=$( df -H )

当我想进一步处理时$a,将换行符替换为一个空格,并且所有行现在都位于一行上,这很难grep进行:


echo $a

避免换行符被命令替换删除的简单技巧是什么?


查看完整描述

3 回答

?
慕圣8478803

我试图解决这个问题,因为在F#脚本上运行解释器时,我使用bash进行流式处理。经过一番反复尝试,最终解决了该问题:


$ cat fsi.ch

#!/bin/bash

echo "$(fsharpi --quiet --exec --nologo $1)"


$ fsi.ch messages.fsx

Welcome to my program. Choose from the menu:

new | show | remove

假设,当然,您需要运行一个终端程序。希望这可以帮助。


查看完整回答
反对 回复 2019-12-27
?
慕粉4167745

注意:read到达EOF时以非零退出。这导致方法2和方法3以非零值退出,因此掩盖了主命令的退出状态。在方法1中,使用main_cmd ; printf x还会覆盖主要命令的退出代码并干扰错误处理。尝试main_cmd && printf x改行(如果尾行仅在成功退出时才重要)或main_cmd ; ec=$?; printf x; exit $ec在成功和错误情况下都保留尾随空格。

查看完整回答
反对 回复 2019-12-27
?
尚方宝剑之说

不追踪的换行符不会被删除

您正在寻找的换行符在那里,只是看不到它们,因为您在使用时echo没有引用变量。


验证:


$ a=$( df -H )

$ echo $a

Filesystem Size Used Avail Use% Mounted on /dev/sda3 276G 50G 213G 19% / udev 2.1G 4.1k 2.1G 1% /dev tmpfs 832M 820k 832M 1% /run none 5.3M 0 5.3M 0% /run/lock none 2.1G 320k 2.1G 1% /run/shm

$ echo "$a"

Filesystem      Size  Used Avail Use% Mounted on

/dev/sda3       276G   50G  213G  19% /

udev            2.1G  4.1k  2.1G   1% /dev

tmpfs           832M  820k  832M   1% /run

none            5.3M     0  5.3M   0% /run/lock

none            2.1G  320k  2.1G   1% /run/shm

尾随换行符被删除

正如@ user4815162342正确指出的那样,尽管不删除输出中的换行符,但使用命令替换来删除结尾的换行符。请参见下面的实验:


$ a=$'test\n\n'

$ echo "$a"

test



$ b=$(echo "$a")

$ echo "$b"

test

$

在大多数情况下,这无关紧要,因为echo将添加删除的换行符(除非使用该-n选项调用该换行符),但是在某些极端情况下,程序输出中的尾随换行符不止一个,并且对于一些理由。


解决方法

1.添加虚拟字符

在这种情况下,如@Scrutinizer所述,您可以使用以下解决方法:


$ a=$(printf 'test\n\n'; printf x); a=${a%x}

$ echo "$a"

test



说明:在换行符之后,将字符x添加到输出中(使用printf x)。由于换行符不再尾随,因此不会被命令替换删除。下一步是x使用中的%运算符删除添加的${a%x}。现在我们有了原始输出,其中包含所有换行符!!!


2.使用进程替换进行读取

代替使用命令替换将程序的输出分配给变量,我们可以使用进程替换 来将程序的输出提供给read内置命令(贷记为@ormaaj)。流程替换将保留所有换行符。将输出读取到变量有点棘手,但是您可以这样做:


$ IFS= read -rd '' var < <( printf 'test\n\n' ) 

$ echo "$var"

test



说明:


我们将read命令的内部字段分隔符设置为null IFS=。否则read,不会将整个输出分配给var,而只会将第一个令牌分配给。

我们read用options 调用-rd ''。的作用r是防止反斜杠充当特殊字符,并将d ''定界符设置为no,以便read读取整个输出,而不仅仅是第一行。

3.从管道读取

代替使用命令或进程替换将程序的输出分配给变量,我们可以将程序的输出通过管道传递给read命令(贷方为@ormaaj)。管道还保留所有换行符。但是请注意,这时候我们设置的lastpipe外壳可选行为,使用了shopt内置。这是必需的,以便read在当前Shell环境中执行命令。否则,变量将在子shell中分配,并且脚本的其余部分将无法访问该变量。


$ cat test.sh 

#!/bin/bash

shopt -s lastpipe

printf "test\n\n" | IFS= read -rd '' var

echo "$var"

$ ./test.sh 

test



$


查看完整回答
反对 回复 2019-12-27

添加回答

回复

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信