tags:

views:

1714

answers:

5

How would I get awk to output $9 $10 $11 etc as some of my files have spaces in them.

ls -l | grep ^- | awk '{print $9}'
+1  A: 

There's probably a better approach that involves combining fields somehow, but:

$ echo 1 2 3 4 5 6 7 8 9 10 11 12 13 14... | awk '{for (i = 9 ; i <= NF ; i++) printf "%s ", $i}'
9 10 11 12 13 14... 

Using printf "%s " $i will print the i-th field with a space after it, instead of a newline. The for loop just says to go from field 9 to the last field.

Mark Rushakoff
+1  A: 
echo 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 | awk 'BEGIN {OFS=ORS=""} ; {for (i=9;i<NF;i++) print $i " "; print $NF "\n"}'
Amro
+5  A: 

A better solution: Don't attempt to parse ls output in the first place.

The official wiki of the irc.freenode.org #bash channel has an explanation of why this is a Bad Idea, and what alternate approaches you can take instead: http://mywiki.wooledge.org/ParsingLs

Use of find, stat and similar tools will provide the functionality you're looking for without the pitfalls (not all of which are obvious -- some occur only when moving to platforms with different ls implementations).

For your specific example, I'm guessing that you're trying to find only files (and not directories) in your current directory; your current implementation using ls -l is buggy, as it excludes files which have +t or setuid permissions. The Right Way to implement this would be the following:

find . -maxdepth 1 -type f -printf '%f\n'
Charles Duffy
Consider the use of \0 in place of \n if you're feeding the output into another command like `grep -z` or `xargs -0`. This makes `-printf` work like `-print0`.
Dennis Williamson
+1  A: 

If you still insist on the ls -l instead of find or other tools, here is my solution. It is not pretty and destructive:

  1. Destroy $1 .. $8 by setting them to "" via a for loop
  2. That leaves a bunch of spaces preceding $9, remove them using the sub() command
  3. Print out the remaining

    ls -l | awk '{for (i = 1; i < 9; i++) $i = ""; sub(/^ */, ""); print}'

Hai Vu
+1  A: 

Just for completion. It can also be done with sed:

# just an exercise in regex matching ...
ls -l | sed -E  -e 1d -e /^[^-]/d -e 's/^([^ ]+ +){8}//'