Recipe 4.19 Program: words

Have you ever wondered how programs like ls generate columns of sorted output that you read down the columns instead of across the rows? For example:

awk      cp       ed       login    mount    rmdir    sum basename csh      egrep    ls       mt       sed      sync cat      date     fgrep    mail     mv       sh       tar chgrp    dd       grep     mkdir    ps       sort     touch chmod    df       kill     mknod    pwd      stty     vi chown    echo     ln       more     rm       su

Example 4-2 does this.

Example 4-2. words
  #!/usr/bin/perl -w   # words - gather lines, present in columns      use strict;      my ($item, $cols, $rows, $maxlen);   my ($xpixel, $ypixel, $mask, @data);      getwinsize( );      # first gather up every line of input,   # remembering the longest line length seen   $maxlen = 1;           while (<>) {       my $mylen;       s/\s+$//;       $maxlen = $mylen if (($mylen = length) > $maxlen);       push(@data, $_);   }      $maxlen += 1;               # to make extra space      # determine boundaries of screen   $cols = int($cols / $maxlen) || 1;   $rows = int(($#data+$cols) / $cols);      # pre-create mask for faster computation   $mask = sprintf("%%-%ds ", $maxlen-1);      # subroutine to check whether at last item on line   sub EOL { ($item+1) % $cols =  = 0 }        # now process each item, picking out proper piece for this position   for ($item = 0; $item < $rows * $cols; $item++) {       my $target =  ($item % $cols) * $rows + int($item/$cols);       my $piece = sprintf($mask, $target < @data ? $data[$target] : "");       $piece =~ s/\s+$// if EOL( );  # don't blank-pad to EOL       print $piece;       print "\n" if EOL( );   }      # finish up if needed   print "\n" if EOL( );      # not portable -- linux only   sub getwinsize {       my $winsize = "\0" x 8;       my $TIOCGWINSZ = 0x40087468;       if (ioctl(STDOUT, $TIOCGWINSZ, $winsize)) {           ($rows, $cols, $xpixel, $ypixel) = unpack('S4', $winsize);       } else {           $cols = 80;       }   }

The most obvious way to print out a sorted list in columns is to print each element of the list, one at a time, padded out to a particular width. Then when you're about to hit the end of the line, generate a newline. But that only works if you're planning on reading each row from left to right. If you instead expect to read it down each column, this approach won't do.

The words program is a filter that generates output going down the columns. It reads all input, keeping track of the length of the longest line seen. Once everything has been read in, it divides the screen width by the length of the longest input record seen, yielding the expected number of columns.

Then the program goes into a loop that executes once per input record, but the output order isn't in the obvious order. Imagine you had a list of nine items:

Wrong       Right -----       ----- 1 2 3       1 4 7 4 5 6       2 5 8 7 8 9       3 6 9

The words program does the necessary calculations to print out elements (1,4,7) on one line, (2,5,8) on the next, and (3,6,9) on the last.

To figure out the current window size, this program does an ioctl call. This works fine on the system it was written for. On any other system, it won't work. If that's good enough for you, then good for you. Recipe 12.17 shows how to find this on your system using the ioctl.ph file, or with a C program. Recipe 15.4 shows a more portable solution, but that requires installing a CPAN module.

4.19.1 See Also

Recipe 15.4



Perl Cookbook
Perl Cookbook, Second Edition
ISBN: 0596003137
EAN: 2147483647
Year: 2003
Pages: 501

flylib.com © 2008-2017.
If you may any questions please contact us: flylib@qtcs.net