coh*_*ell 5 command-line bash pipe stdout
我正在使用 lolcat 来获取 ls 的彩色输出。为此,我已将 /usr/bin/ls 复制到 /usr/bin/lsslss(以避免由于别名无法接受 $* 或 $@ 而导致无限循环),并且我添加了该函数:
ls(){ lsslss $* | lolcat; }
到 .bashrc
问题是,当我使用 ls 时,管道一次通过管道传输每个文件,因此它显示为一个长列表,如下所示:
而不是这样的表:
要将其更改为表,我可以将输出通过管道传输到列命令中。但是当我这样做时,它又变回了一个长列表(可能是因为列只对其进行格式化而不是将其更改为行)
我原本打算这样做:
ls(){ lsslss $* | columns | lolcat; }
无论如何,我想知道有没有办法通过管道传输原始输出而不是使用 | 能够将列的输出通过管道传输到 lolcat 中?
提前致谢。抱歉,如果我的问题措辞不当或难以理解。我几乎总能找到已经问过的问题,所以我不经常发布问题。
Byt*_*der 14
扩展@dessert 的答案,您需要做更多工作才能使您的彩色ls版本ls在(希望?)所有情况下都表现得像真实的一样。问题是它ls并不意味着要被解析,而只是为了人眼。为此,它会根据环境强烈调整其工作方式,例如它是连接到终端还是输出到管道。
首先,您不需要单独的/bin/lsslss可执行文件来避免递归。使用内置的shellcommand从磁盘调用可执行文件,忽略任何 shell 函数或同名别名。
其次,$*将所有函数参数作为单个字符串提供,然后由于未加引号而进行分词。如果您有带空格的参数,这可能会产生令人惊讶的错误结果。始终使用"$@",它完全保留最初给出的所有参数,没有连接或进一步拆分。
第三,根据您放置定义的位置,ls () { ... ;}如果ls已经是别名,则定义函数的语法可能不起作用,因为别名扩展首先发生,从而导致语法错误。通过function在它之前写入来使用显式语法。
然后,我们可以使用ls'-C标志手动启用列输出:
function ls() { command ls -C "$@" | lolcat ;}
Run Code Online (Sandbox Code Playgroud)
但是,如果我们只是这样做并通过管道输出lolcat(或其他任何东西),您会注意到它不再使用终端的整个宽度,而最多只有 80 列。这是因为如果它的标准输出不再直接连接到它,它就无法检测到终端宽度。不过,您的 shell 仍然知道终端,并且它会COLUMNS用它检测到的宽度填充变量。但是,由于默认情况下不导出此变量,ls因此看不到此值。我们可以仅针对此命令手动传递它,例如:
function ls() { COLUMNS="$COLUMNS" command ls -C "$@" | lolcat ;}
Run Code Online (Sandbox Code Playgroud)
现在我们应该总是得到正确的宽度,但是如果我们真的想ls通过其他东西来管道怎么办?通常你不应该这样做,因为正如我在开始时所说的,ls永远不应该被解析。有时它可能仍然有用(有些脚本可能会遗憾地依赖它),所以让我们尝试至少保留原始行为。现在,我们总是将列作为 eg 的输出ls | cat。(它在那里不再着色,因为lolcat还会检查它是否输出到终端或管道并在后一种情况下关闭颜色)
让我们为我们的函数添加一个检查,ls如果它是管道,则使用普通实数,而我们的花式彩虹列版本仅用于终端视图。标准输出(文件描述符 1)是否是终端/TTY 可以简单地检查[[ -t 1 ]]:
function ls() {
if [[ -t 1 ]] ; then COLUMNS="$COLUMNS" command ls -C "$@" | lolcat ; else command ls "$@" ; fi
}
Run Code Online (Sandbox Code Playgroud)
我认为这应该足以捕捉到所有ls预期的特殊/不同行为的情况,以便您的函数仅在直接在终端中查看时添加颜色,否则不会改变任何内容。
| 归档时间: |
|
| 查看次数: |
1140 次 |
| 最近记录: |