4

我很好奇这个看似复杂的任务是否有一个简单的解决方案:

假设我有这个文件:

lineone
linetwo
linethree
linefour

lineone
linetwo
linethree
linefour

lineone
lineone
lineone
linetwo
linethree
linefour

您是否可以想到一种简单的方法来获得此结果:

lineone[1]
linetwo[1]
linethree[1]
linefour[1]

lineone[2]
linetwo[2]
linethree[2]
linefour[2]

lineone[3]
lineone[4]
lineone[5]
linetwo[3]
linethree[3]
linefour[3]

简而言之,是否有一个简单的 bash 算法可以让我按照在文件中出现的顺序对每一行进行编号?如果没有一些涉及额外存储的复杂解决方案,我无法找到一种方法,而且我用于换行的常用命令在这里毫无用处。sed 只会用相同的值替换所有出现的行,而字符串替换( ${string/substring/replace} )对我没有帮助,因为它不会保持行的顺序。

非常感激。

4

3 回答 3

3

以下awk命令通过保留行数组及其计数来工作。

awk '{if($0~/./){a[$0]++;print $0"["a[$0]"]";} else print}' file

例子:

$ cat file
lineone
linetwo
linethree
linefour

lineone
linetwo
linethree
linefour

lineone
linetwo
linethree
linefour

lineone
lineone
lineone
linetwo
linethree
linetwo

$ awk '{if($0~/./){a[$0]++;print $0"["a[$0]"]";} else print}' file
lineone[1]
linetwo[1]
linethree[1]
linefour[1]

lineone[2]
linetwo[2]
linethree[2]
linefour[2]

lineone[3]
linetwo[3]
linethree[3]
linefour[3]

lineone[4]
lineone[5]
lineone[6]
linetwo[4]
linethree[4]
linetwo[5]
于 2012-12-20T16:55:09.563 回答
2

一个快速的 100% 纯 bash 答案:

#!/bin/bash

declare -A lines=()
while read -r l; do
    if [[ -n "$l" ]]; then
        echo "$l[$((++lines[$l]))]"
    else
        echo "$l"
    fi
done < file.txt

根据需要,这是一个简单的 bash 算法,它可以让 [您] 按照在文件中出现的顺序对每个 [非空] 行进行编号

这不是最快的方法(awk答案更快更有效,但awk答案不是bash 算法)。

诀窍是使用关联数组lines,其键是文件的行,并在每次读取非空行时递增相应键的值。

或单线以打动您的祖母:

declare -A lines=(); while read -r l; do [[ -n "$l" ]] && echo "$l[$((++lines[$l]))]"; || echo "$l"; done < file.txt
于 2012-12-20T17:20:59.727 回答
2

一个神秘的 Perl 单线:

perl -00 -lpe 's/$/"[". $. . "]"/gem'

我看到我根据模棱两可的例子误解了这个问题。这是 Perl 代码段,用于对所描述的行进行编号:

perl -lpe '/\S/ and $_ .= "[" . ++$n{$_} . "]"'
于 2012-12-20T21:17:34.650 回答