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