views:

1441

answers:

7

I've seen some horrific code written in Perl but I can't make head or tail of this one.

It's in some networking code that we use to communicate with a server and I assume it's something to do with buffering (since it sets $|).

But I can't figure out why there's multiple selects or the array reference. Can anyone help me out?

+41  A: 

It's a nasty little idiom for setting autoflush on a filehandle other than STDOUT.

select() takes the supplied filehandle and (basically) replaces STDOUT with it, and it returns the old filehandle when it's done.

So (select($s),$|=1) redirects the filehandle (remember select returns the old one), and sets autoflush ($| = 1). It does this in a list ((...)[0]) and returns the first value (which is the result of the select call - the original STDOUT), and then passes that back into another select to reinstate the original STDOUT filehandle. Phew.

But now you understand it (well, maybe ;)), do this instead:

use IO::Handle;
$fh->autoflush;
Dan
Nasty, why exactly?
paxdiablo
@Pax: why? LOOK AT IT!
Dan
Sorry, I thought you meant nasty as in dodgy-functionality rather than lack-of-clarity. So it's passing the result of the inner select to the outer select, reselecting the original. That makes sense now but you're right, I'll probably scrap it and use autoflush.
paxdiablo
I wrote that ugly piece of code long before IO::Handle->autoflush existed. Please let it die a quiet death. :)
Randal Schwartz
@Randal: So *you're* to blame! My apologies for describing it as 'nasty', in the context of IO::Handle->autoflush not existing then it is indeed a crafty little chunk of code ;) (/me backpedals furiously)
Dan
I don’t see any reason to use the Lisp-ish version. Even in absence of IO::Handle you can write it much more nicely by using `for` to topicalise the old handle for re-selection. See my answer.
Aristotle Pagaltzis
+8  A: 

It's overly clever code for turning on buffer flushing on handle s and then re-selecting the current handle.

See perldoc -f select for more.

Andy Lester
+20  A: 

The way to figure out any code is to pick it apart. You know that stuff inside parentheses happens before stuff outside. This is the same way you'd figuring out what code is doing in other languages.

The first bit is then:

( select(s), $|=1 )

That list has two elements, which are the results of two operations: one to select the s filehandle as the default then one to set $| to a true value. The $| is one of the per-filehandle variables which only apply to the currently selected filehandle (see Understand global variables at The Effective Perler). In the end, you have a list of two items: the previous default filehandle (the result of select), and 1.

The next part is a literal list slice to pull out the item in index 0:

( PREVIOUS_DEFAULT, 1 )[0]

The result of that is the single item that is previous default filehandle.

The next part takes the result of the slice and uses it as the argument to another call to select

 select( PREVIOUS_DEFAULT );

So, in effect, you've set $| on a filehandle and ended up back where you started with the default filehandle.

brian d foy
+5  A: 

In another venue, I once proposed that a more comprehensible version would be thus:

for ( select $fh ) { $| = 1; select $_ }

This preserves the compact idiom’s sole advantage that no variable needs be declared in the surrounding scope.

Or if you’re not comfortable with $_, you can write it like this:

for my $prevfh ( select $fh ) { $| = 1; select $prevfh }

The scope of $prevfh is limited to the for block. (But if you write Perl you really have no excuse to be skittish about $_.)

Aristotle Pagaltzis
Both yours and Randall's versions are less than obvious, but at least Randall's has the advantage of being short.. Is there some reason why my $old_fh = select($fh); $| = 1; select($old_fh); doesn't work? As if it's comprehensibility you're going for it would seem a considerably more sensible choice
Dan
Did you see any claims about it not working? In any case if you actually want a sensible choice you’ll use IO::Handle. As for shortness, Randall’s has a 1–4 character advantage depending on whitespace.
Aristotle Pagaltzis
That was a completely honest question, by the way. I've never been in a position where I've had to do it (at least where IO::Handle hasn't been available).
Dan
So go ahead and use it. I didn’t claim that my solution was the best nor that any other solution doesn’t work; I merely offered, in the spirit of TMTOWTDI, a Perl-ish rather than Lisp-ish idiom.
Aristotle Pagaltzis
+1  A: 

please check perldoc -f select. For the meaning of $|, please check perldoc perlvar

ghostdog74
+17  A: 
select($fh)

Select a new default file handle. See http://perldoc.perl.org/functions/select.html

(select($fh), $|=1)

Turn on autoflush. See http://perldoc.perl.org/perlvar.html

(select($fh), $|=1)[0]

Return the first value of this tuple.

select((select($fh), $|=1)[0])

select it, i.e. restore the old default file handle.


Equivalent to

$oldfh = select($fh);
$| = 1;
select($oldfh);

which means

use IO::Handle;
$fh->autoflush(1);

as demonstrated in the perldoc page.

KennyTM
If someone doesn't want to use `IO::Handle` the least they could do is wrap this monstrosity in a function like `sub flush ($) { select((select($_[0]), $|=1)[0]); }`
Chris Lutz
If you're going to put it in a function then you might as well write it the long way with the temporary to hold the old handle. :)
hobbs
@hobbs - Eh, I could go either way. For functions that simple, once it's in a function, you never really have to look at it again, nor is the functionality of a `flush` function ever likely to change. Perhaps the horrific version is slightly faster. Perhaps you can use it to scare an intern.
Chris Lutz
+1  A: 

It is overoptimization to skip loading IO::Handle.

use IO::Handle;
$fh->autoflush(1);

is much more readable.

Alexandr Ciornii
I wouldn't say overoptimization. using lexical filehandles is a *relatively* new thing
Nathan Fellman
@Nathan Fellman: in the "only a decade" sense of new, anyway...
hobbs
true. I work with a codebase that has a number of scripts from back then. Until recently I maintained things written in perl 4.
Nathan Fellman
@Nathan Fellman: it can be used with non-lexical filehandles too.
Alexandr Ciornii
I didn't know that! Thanks!
Nathan Fellman