使用 tput 时如何在 bash 中使用 printf 格式化列

How to format columns with printf in bash when tput is used

使用 tput 更改字符串的长度,因此列未对齐。如何解决这个问题?

在 bash 脚本中尝试了以下代码。

B="$(tput bold)" # Bold text
N="$(tput sgr0)" # Normal text

function testing(){

IN="KEYS | VALUES | DESCRIPTION
id | ${B}10${N} or ${B}20${N} | Enter ID. Default is ${B}10${N}.
status | ${B}true${N} or ${B}false${N} | Enter status of something. Default is ${B}true${N}.
style | Example: Standard | Give suitable standard."

    IFS= 
    while read -r lines; do
        IFS='|' read -r -a array <<< "$lines"
        printf "%-35s %-35s %-s\n" "${array[0]}" "${array[1]}" "${array[2]}"
    done <<< "$IN"

    read -p "$*"
    exit 0
}

输出类似于:

    KEYS          VALUES                              DESCRIPTION
    id            **10** or **20**           Enter ID. Default is **10**.
    status        **true** or **false**      Enter status of something. Default is **true**.
    style         Example: Standard                   Give suitable standard.

预计:

    KEYS          VALUES                              DESCRIPTION
    id            **10** or **20**                    Enter ID. Default is **10**.
    status        **true** or **false**               Enter status of something. Default is **true**.
    style         Example: Standard                   Give suitable standard.

正如我在评论中提到的:

  • 当我 运行 我的 bash 环境中的代码时,我发现 2 个有问题的 DESCRIPTION 字符串丢失了 20 个缩进空格
  • 变量 (${#B} & ${#N}) 的长度分别为 4 个和 6 个字符
  • 这些变量在 VALUES 字段中出现两次,总共 20 个(不可打印的)字符
  • printf/%s 将 20 个不可打印的字符计为输出的一部分,因此看起来 20 个空格是 'lost'(即,printf 打印 35 个字符...只是其中 20 个字符是非打印的)

如果我们测量第二个字段的长度 - 有和没有特殊字符 - 然后使用差异(即不可打印字符的数量)来增加printf命令的第二个字段的格式宽度?

我们将第二个字段(包括我们的特殊字符)的长度存储在一个新变量中 len2:

len2="${#array[1]}"

接下来我们要去除特殊字符并测量结果字符串的长度并放入变量lenx:

x="${array[1]//${B}/}"    # strip out all occurrences of variable 'B'
x="${x//${N}/}"           # strip out all occurrences of variable 'N'
lenx=${#x}

注意:我在让 trsed 正确删除特殊字符时遇到了一些问题;我愿意接受建议。从好的方面来说......我没有产生任何子进程。

我们会将新的格式宽度存储在变量 w2 中('w'idth for field '2'),如下所示:

w2=$(( 35 + len2 - lenx ))

而新的 printf 格式字符串变为:

printf "%-35s %-${w2}s %-s\n" ...

综合起来我们得到:

B="$(tput bold)" # Bold text
N="$(tput sgr0)" # Normal text

function testing(){

IN="KEYS | VALUES | DESCRIPTION
id | ${B}10${N} or ${B}20${N} | Enter ID. Default is ${B}10${N}.
status | ${B}true${N} or ${B}false${N} | Enter status of something. Default is ${B}true${N}.
style | Example: Standard | Give suitable standard."

    IFS= 
    while read -r lines; do
        IFS='|' read -r -a array <<< "$lines"

        len2="${#array[1]}"

        x="${array[1]//${B}/}"
        x="${x//${N}/}"
        lenx=${#x}

        w2=$(( 35 + len2 - lenx ))
   #    echo "w2 = ${w2}"

        printf "%-35s %-${w2}s %-s\n" "${array[0]}" "${array[1]}" "${array[2]}"
    done <<< "$IN"

    read -p "$*"
    exit 0
}

运行 我的 bash 环境中的脚本生成:

$ testing
KEYS                                 VALUES                              DESCRIPTION
id                                   10 or 20                            Enter ID. Default is 10.
status                               true or false                       Enter status of something. Default is true.
style                                Example: Standard                   Give suitable standard.

注意:粗体字段在我的终端上打印为 bold ...它们只是不 copy/display 打印为 bold 在上面的回答中。

如果您取消注释行 - echo "w2 = ${w2}" - 您应该会发现(对于感兴趣的两行)我们将为第二个字段使用 55 的格式宽度(即, 所需的 35 加上额外的 20 以补偿 20 个字符的不可打印字符)。