I know that with grep I can use the fields -A
and -B
to pull previous and next lines from a match.
However they pull in all lines between the match based on however many lines are specified.
grep -r -i -B 5 -A 5 "match"
I'd like to only receive the 5th line before a match and the 5th line after the match in addition to the matched line and not get the lines between.
Is there a way to do this with the grep
?
If:
Then:
This is basically Glenn's solution, but implemented with Bash, Grep, and sed.
Note that line numbers less than 1 will make sed error, and line numbers greater than the number of lines in the file will make it print nothing.
This is just the bare minimum. To make it work recursively and handle the above line number cases would take some doing.
It can't be done with only
grep
. Ifed
's an option:The script basically says: for every match of /match/, print the line 5 lines before that, then 5 lines after that, then 5 lines after that.
Here we are using awk's
system(command)
function to call externalsed
command to print the lines which awk matched with patternmatch
with 5th lines before and after the match.The syntax is easy, you just need to put the external command itself inside double-quote as well as its switches and escape the things you want exactly pass to the command, everything else related to the
awk
itself options should be outside of the quotes. So the below sed:translate into:
NR
is the line number that matched with the patternmatch
andFILENAME
is the of current processing filename passing byawk
.using @glenn's example text file and using perl instead of awk:
will give the same results, but running faster:
The tool you want to use is called sift. This is basically a grep on steroids. Grep in parallel. Sift has a huge amount of options to do exactly what you want - specifically to return a particular line relative to a match(s) which may/may not be followed by /preceded by some text.
It amazes me that sift is not mainstream gnu as it was written in the go language but installs on Linux just fine. IT searches in parallel using all cpus huge quantities of text where grep just takes weeks to do the same.
Sift website - see examples