ls -l 如何如此整齐地格式化它的输出?

问题描述

ls -l 如何格式化其输出以使列内容整齐排列?

最佳方法

ls 的源代码可在 GNU Savannah 上在线浏览。在大多数情况下,计算所需的最大宽度(例如,对文本使用 mbswidth 函数),然后使用经典的 C printf 函数格式说明符和一些手动填充。例如,参见函数 format_user_or_group()gobble_file()

长话短说:没有 “magic”,只有大量的计算。


如果您想为自己的输出提供如此整洁的表格,请使用 column

$ grep -vE '^#' /etc/fstab
UUID=cdff3742-9d03-4bc1-93e3-ae50708474f2 /               ext4    errors=remount-ro 0       1
/dev/mapper/lvmg-homelvm /home           btrfs   defaults,compress=lzo,space_cache,relatime 0       2
UUID="bb76cd0d-ae1d-4490-85da-1560c32679cd" none    swap sw 0 0
UUID="a264b1b1-cf82-40aa-ab9e-a810cfba169a" /home/muru/arch     btrfs defaults,compress=lzo,space_cache,relatime 0       2

$ grep -vE '^#' /etc/fstab | column -t
UUID=cdff3742-9d03-4bc1-93e3-ae50708474f2    /                ext4   errors=remount-ro                           0  1
/dev/mapper/lvmg-homelvm                     /home            btrfs  defaults,compress=lzo,space_cache,relatime  0  2
UUID="bb76cd0d-ae1d-4490-85da-1560c32679cd"  none             swap   sw                                          0  0
UUID="a264b1b1-cf82-40aa-ab9e-a810cfba169a"  /home/muru/arch  btrfs  defaults,compress=lzo,space_cache,relatime  0  2

次佳方法

除了 @muru ‘s answer 之外,这里是 source code 的一部分,它计算 width 以实现输出的右对齐。 :

static void
format_user_or_group (char const *name, unsigned long int id, int width)
{
  size_t len;

  if (name)
    {
      int width_gap = width - mbswidth (name, 0);
      int pad = MAX (0, width_gap);
      fputs (name, stdout);
      len = strlen (name) + pad;

      do
        putchar (' ');
      while (pad--);
    }
  else
    {
      printf ("%*lu ", width, id);
      len = width;
    }

  dired_pos += len + 1;
}

它使用 printf ("%*lu ", width, id); 。注意:可变字段宽度说明符“*”

在这种情况下,无法预测在执行 ls -l 时我们需要多大的字段宽度,即,目录名称的长度可能会有所不同。这意味着字段宽度本身需要是一个变量,程序将为其计算一个值。

C 在字段宽度说明符的位置使用星号向 printf 指示它将找到包含字段宽度值作为附加参数的变量。

例如,假设 width 的当前值为 5。语句:

printf ("%*d%*d\n", width, 10, width, 11);

将打印:(注意间距)

   10   11

参考资料

  • How does ls -l format its output so neatly?

ls -l 如何如此整齐地格式化它的输出?

扫一扫手机访问