Differences between revisions 9 and 21 (spanning 12 versions)
Revision 9 as of 2007-10-16 13:55:46
Size: 1070
Editor: p54BFE162
Comment: make cut work (with tabs in the file cut -f2 would only give part of the desired output) -- shouldn't this example be removed?! -- so much prettier in sed/awk.
Revision 21 as of 2011-03-01 15:14:40
Size: 1803
Editor: sn18
Comment:
Deletions are marked like this. Additions are marked like this.
Line 1: Line 1:
[[Anchor(faq11)]]
== How can I print the n'th line of a file? ==
<<Anchor(faq11)>>

= How can I print the n'th line of a file? =
Line 4: Line 5:
Line 7: Line 9:
Line 11: Line 12:
Line 14: Line 16:
Line 18: Line 19:
Line 19: Line 21:
   head -n $n $file | tail -n 1     head -n $n $file | tail -n 1
Line 21: Line 23:
Another approach, using [[AWK]]:
Line 22: Line 25:
Another approach, using ["AWK"]:
Line 26: Line 28:
Line 30: Line 31:
   x=3 y=4;    x=3 y=4
Line 32: Line 33:
   head -n $y "$file" | tail -n $(($y - $x + 1)) # Same    head -n $y "$file" | tail -n $((y - x + 1))   # Same
   head -n $y "$file" | tail -n +$x # If your tail supports it
Line 35: Line 37:

Using nl(1) (and grep(1) (and cut(1))):
In Bash 4, this can also be done with mapfile builtin:
Line 39: Line 40:
   nl -b a file | grep '^ *<number of line>\W' | cut -f 1 --complement    mapfile -ts $((n-1)) -n 1
   echo "$MAPFILE"
Line 41: Line 43:
By changing the value of argument for -n, you can get more than one line in the array MAPFILE.

=== Note ===
In most cases, you should sanitize your variable n to be sure, that it's not containing any of non-digits, before feeding it to sed or awk. You can do it with such simple code:

{{{
   # Bash
   n=${n//[!0-9]/}

   # POSIX
   n=$(printf "%s" "$n"|tr -cd '0-9')
}}}
 . One can argue that an `n` value of `Five brown horses with 3 feet and 9 little piggies.` should yield an error rather than silently behave as though it was really `39`. If you want to code safe, quote your expansions and be done with it. --[[Lhunath]]

----
CategoryShell

How can I print the n'th line of a file?

The dirty (but not quick) way would be:

    sed -n ${n}p "$file"

but this reads the whole input file, even if you only wanted the third line.

This one avoids that problem:

    sed -n "$n{p;q;}" "$file"

At line $n the command "p" is run, printing it, with a "q" afterwards: quit the program.

Another way, more obvious to some, is to grab the last line from a listing of the first n lines:

   head -n $n $file | tail -n 1

Another approach, using AWK:

   awk "NR==$n{print;exit}" file

If you want more than one line, it's pretty easy to adapt any of the previous methods:

   x=3 y=4
   sed -n "$x,${y}p;${y}q;" "$file"                # Print lines $x to $y; quit after $y.
   head -n $y "$file" | tail -n $((y - x + 1))     # Same
   head -n $y "$file" | tail -n +$x                # If your tail supports it
   awk "NR>=$x{print} NR==$y{exit}" "$file"        # Same

In Bash 4, this can also be done with mapfile builtin:

   mapfile -ts $((n-1)) -n 1
   echo "$MAPFILE"

By changing the value of argument for -n, you can get more than one line in the array MAPFILE.

Note

In most cases, you should sanitize your variable n to be sure, that it's not containing any of non-digits, before feeding it to sed or awk. You can do it with such simple code:

   # Bash
   n=${n//[!0-9]/}

   # POSIX
   n=$(printf "%s" "$n"|tr -cd '0-9')
  • One can argue that an n value of Five brown horses with 3 feet and 9 little piggies. should yield an error rather than silently behave as though it was really 39. If you want to code safe, quote your expansions and be done with it. --Lhunath


CategoryShell

BashFAQ/011 (last edited 2020-05-07 08:35:17 by intranet)