我有以下文件(请注意,它们========
实际上存在于文件中):
start ======== id: 5713
start ======== id: 5911
start ======== id: 5911
end ========= id: 5911
start ======== id: 6111
end ========= id: 5713
start ======== id: 31117
我想删除具有相同 id 且分别具有 和 的任何两start
行end
。
基于上面的例子,输出将是:
start ======== id: 5911
start ======== id: 6111
start ======== id: 31117
如何用bash
, awk
, sed
... 做到这一点?
答案1
在每个 Unix 机器上的任何 shell 中使用任何 awk,这将打印输入中存在的尽可能多的不成对的开始和/或结束语句:
$ cat tst.awk
$1 == "start" { beg[$NF] = $0; delta = 1 }
$1 == "end" { end[$NF] = $0; delta = -1 }
{ cnt[$NF] += delta }
END {
for ( key in cnt ) {
for (i=1; i<=cnt[key]; i++) {
print beg[key]
}
for (i=-1; i>=cnt[key]; i--) {
print end[key]
}
}
}
$ awk -f tst.awk file
start ======== id: 5911
start ======== id: 6111
start ======== id: 31117
为了更好地演示使用更全面的示例输入:
$ cat file
start ======== id: 5713
start ======== id: 5911
start ======== id: 5911
start ======== id: 5911
end ========= id: 5911
start ======== id: 6111
end ========= id: 5713
end ========= id: 5713
start ======== id: 31117
$ awk -f tst.awk file
end ========= id: 5713
start ======== id: 5911
start ======== id: 5911
start ======== id: 6111
start ======== id: 31117
答案2
只需 sed 和 nl 并排序:
nl <filename> -s ":"|sort -t ":" -k 3 -k 2 | sed -n ":x s/\n[0-9 ]*$//;/end[^\n]*$/{N;bx};s/\(.*\)[ 0-9]*:end .*id:\( [0-9]*\).*\n.*start.*id:\2[^0-9]*$/\1/;tx;s/\n$//;/start/{P;D};/^[ 0-9]*:end[^\n]*/{s/\n[0-9:]*$/$/;N;bx};/start/P;/end/P;" | sort -n| sed "s/[ 0-9]*://"
nl tt -s ":"|sort -t ":" -k 3 -k 2 | sed -n ":x s/\n[0-9 ]*$//;/end[^\n]*$/{N;bx};s/\(.*\)[ 0-9]*:end .*id:\( [0-9]*\).*\n.*start.*id:\2[^0-9]*$/\1/;tx;s/\n$//;/start/{P;D};/^[ 0-9]*:end[^\n]*/{s/\n[0-9:]*$/$/;N;bx};/start/P;/end/P;" | sort -n| sed "s/[ 0-9]*://"
start ======== id: 5713
start ======== id: 5713
start ======== id: 5713
start ======== id: 5713
start ======== id: 5911
start ======== id: 6111
end ======== id: 31117
如果顺序不重要(并且每个末端都有起始线):
sort <filename> -t ":" -k 2|sed -e '/end/{N;d;}
start ======== id: 31117
start ======== id: 5911
start ======== id: 6111
这更好(需要修复,但可以工作):
sort <filename> -t ":" -k 2 | sed -n ":x ;/end[^\n]*$/{N;bx};s/\(.*\)end .*id:\( [0-9]*\).*start.*id:\2[^0-9]*$/\1/;tx;s/\n$//;/start/{P;D};/^end[^\n]*/{s/\n$/$/;N;bx};/start/P;/end/P"
cat tt
start ======== id: 5713
start ======== id: 5713
start ======== id: 5713
start ======== id: 5713
start ======== id: 5713
start ======== id: 5713
start ======== id: 5713
dggdgtfZZ
start ======== id: 5713
start ======== id: 5713
start ======== id: 5911
start ======== id: 5911
end ========= id: 5911
start ======== id: 6111
end ========= id: 5713
end ========= id: 5713
end ========= id: 5713
end ========= id: 5713
end ========= id: 5713
start ======== id: 31117
end ======== id: 31117
end ======== id: 31117
sort -t ":" -k 2 tt| sed -n ":x ;/end[^\n]*$/{N;bx};s/\(.*\)end .*id:\( [0-9]*\).*start.*id:\2[^0-9]*$/\1/;tx;s/\n$//;/start/{P;D};/^end[^\n]*/{s/\n$/$/;N;bx};/start/P;/end/P"
end ======== id: 31117
start ======== id: 5713
start ======== id: 5713
start ======== id: 5713
start ======== id: 5713
start ======== id: 5911
start ======== id: 6111