Grep如何抓取前一行後兩行(for,xargs,sed,awk,{},$) - Linux

Table of Contents

----------------------------------------------------------------

data.txt內容是
111
AAAAA
333
444
555
111
AAAAA
333
444
555

希望產出結果是
111
AAAAA
333
444
111
AAAAA
333
444

----------------------------------------------------------------

[問題一]

使用Shell Script來寫(for搭配sed) 正確
greplinenumbers=`grep -E "*AAA*" -n data.txt | cut -d : -f 1`
for getline in $greplinenumbers
do
sed -ne "$(($getline-1)) p" \
-ne "$getline p" \
-ne "$(($getline+1)) p" \
-ne "$(($getline+2)) p" data.txt
done

使用Shell Script來寫(for搭配awk) 錯誤
greplinenumbers=`grep -E "*AAA*" -n data.txt | cut -d : -f 1`
for getline in $greplinenumbers
do
awk "NR==$(($getline-1)) {printf $0"\n"}
NR==$getline {printf $0"\n"}
NR==$(($getline+1)) {printf $0"\n"}
NR==$(($getline+2)) {printf $0"\n"}" data.txt
done

----------------------------------------------------------------

[問題二]

一行指令(xargs搭配awk) 正確
grep -E "*AAA*" -n data.txt | cut -d : -f 1 | xargs -i \
awk 'NR=={}-1 {printf $0"\n"}
NR=={} {printf $0"\n"}
NR=={}+1 {printf $0"\n"}
NR=={}+2 {printf $0"\n"}' data.txt

一行指令(xargs搭配sed) 錯誤
grep -E "*AAA*" -n data.txt | cut -d : -f 1 | xargs -i \
sed -ne '{}-1 p' -ne '{} p' -ne '{}+1 p' -ne '{}+2 p' data.txt

----------------------------------------------------------------

請問大家以上兩個錯誤(紅字部分)應該怎麼修正呢?

: : : 不好意思...我使用的是UNIX系統
: : : 我想請問一下...
: : : 我記得抓關鍵字"AAA"..單行的指令應該是
: : : grep "AAA" data.txt > result.txt
: : : 會搜尋data.txt內的"AAA"的那行關鍵字...並且存在result.txt
: : : 但是我希望能夠取這關鍵字的前一行...和後兩行...(包含AAA的關鍵字那行)
: : : 總共需要四行...請問要如何用...
: : : 有網友提供過...加入-B1 -A2參數...可以達成...
: : : 但是我測試過 grep -B1 -A2 "*AAA*" data.txt > result.txt
: : : 但是系統卻顯示無法辨識...-B1 -A2...
: : : 而且我測試 man grep 也找不到..(但可能我英文不好..)_
: : : 想請各位網友...在提供一下...是否有其他方法...
: : : 還是有其他指令可以達成呢 ...謝謝
: : 不好意思...重新提我一年前問過的問題
: : 因為我目前還沒找到答案
: : 我是用SunOS5.10
: : 指令好像沒有支援grep -B1 -A2
: : 有其他方式嗎
: : awk或是sed有辦法達到這功能嗎
:
: grep -B1 -A2 "*AAA*" data.txt > result.txt
:
: 參考看看 grepab.sh
: ----------------------沿虛線剪下-----------------------------
: greplinenumbers=`grep -E "*AAA*" -n data.txt | cut -d : -f 1`
: rm displaylines.txt > null 2>&1
: for getline in $greplinenumbers
: do
: expr $getline - 1 >> displaylines.txt
: expr $getline >> displaylines.txt
: expr $getline + 1 >> displaylines.txt
: expr $getline + 2 >> displaylines.txt
: done
:
: displaylinenumbers=`cat displaylines.txt`
: rm result.txt > null 2>&1
: for getline in $displaylinenumbers
: do
: sed -n "$getline p" data.txt >> result.txt
: done
: ----------------------沿虛線剪下-----------------------------
:
:
: [mary@localhost ~]$ cat data.txt
: 111
: AAAAA
: 333
: 444
: 555
: 111
: AAAAA
: 333
: 444
: 555
: [mary@localhost ~]$ sh grepab.sh
: [mary@localhost ~]$ cat result.txt
: 111
: AAAAA
: 333
: 444
: 111
: AAAAA
: 333
: 444
: [mary@localhost ~]$
:

--

All Comments

Zanna avatarZanna2011-12-10
問題二 你的引號沒有否配....
Donna avatarDonna2011-12-10
抱歉打錯了. 不過雙引號改單引號, {}也是無法做 加減
Zenobia avatarZenobia2011-12-14
xargs -t -i ... 看一下xargs是以怎樣的情況去執行的...
Emma avatarEmma2011-12-16
sed -ne 2-1 p 未知的指令 - , 那該怎麼讓它 {} 做加減呢?
Agnes avatarAgnes2011-12-20
awk 的 NR=={}-1 , 可以讓它 {} 直接做加減 .
Leila avatarLeila2011-12-24
你試著用xargs 跑bash 讓bash 執行 sed
Una avatarUna2011-12-25
bash -c 'sed -ne "$(({}-1)),$(({}+2)) p" data.txt'
Ophelia avatarOphelia2011-12-27
f大, 多謝, 可以了. PS:上面那個e也可以拿掉.
Frederic avatarFrederic2011-12-28
問題一已解, 自問自答
John avatarJohn2011-12-29
awk 'NR=='"$(($getline-1))"' {printf $0"\n"}
...
Daph Bay avatarDaph Bay2012-01-01
NR=='"$(($getline+2))"' {printf $0"\n"}' data.txt