Perl: the Markov chain saw | |
PerlMonks |
perlfunc:openby gods (Initiate) |
on Aug 24, 1999 at 22:43 UTC ( [id://315]=perlfunc: print w/replies, xml ) | Need Help?? |
openSee the current Perl documentation for open. Here is our local, out-dated (pre-5.6) version: open - open a file, pipe, or descriptor
open FILEHANDLE,EXPR open FILEHANDLE
Opens the file whose filename is given by EXPR, and associates it with FILEHANDLE. If FILEHANDLE is an expression, its value is used as the name of the real filehandle wanted. If EXPR is omitted, the scalar variable of the same name as the FILEHANDLE contains the filename. (Note that lexical variables--those declared with my()--will not work for this purpose; so if you're using my(), specify EXPR in your call to open.)
If the filename begins with
The prefix and the filename may be separated with spaces. These various prefixes correspond to the
If the filename begins with
Opening
If you're unfortunate enough to be running Perl on a system that
distinguishes between text files and binary files (modern operating systems
don't care), then you should check out binmode for tips for dealing with this. The key distinction between systems that
need binmode()
and those that don't is their text file formats. Systems like Unix, MacOS, and Plan9, which delimit lines with a single character, and which encode that character in
C as
When opening a file, it's usually a bad idea to continue normal execution if the request failed, so open() is frequently used in connection with die(). Even if die() won't do what you want (say, in a CGI script, where you want to make a nicely formatted error message (but there are modules that can help with that problem)) you should always check the return value from opening a file. The infrequent exception is when working with an unopened filehandle is actually what you want to do. Examples:
$ARTICLE = 100; open ARTICLE or die "Can't find article $ARTICLE: $!\n"; while (<ARTICLE>) {...
open(LOG, '>>/usr/spool/news/twitlog'); # (log is reserved) # if the open fails, output is discarded
open(DBASE, '+<dbase.mine') # open for update or die "Can't open 'dbase.mine' for update: $!";
open(ARTICLE, "caesar <$article |") # decrypt article or die "Can't start caesar: $!";
open(EXTRACT, "|sort >/tmp/Tmp$$") # $$ is our process id or die "Can't start sort: $!";
# process argument list of files along with any includes
foreach $file (@ARGV) { process($file, 'fh00'); }
sub process { my($filename, $input) = @_; $input++; # this is a string increment unless (open($input, $filename)) { print STDERR "Can't open $filename: $!\n"; return; }
local $_; while (<$input>) { # note use of indirection if (/^#include "(.*)"/) { process($1, $input); next; } #... # whatever } }
You may also, in the Bourne shell tradition, specify an
EXPR beginning with
#!/usr/bin/perl open(OLDOUT, ">&STDOUT"); open(OLDERR, ">&STDERR");
open(STDOUT, ">foo.out") || die "Can't redirect stdout"; open(STDERR, ">&STDOUT") || die "Can't dup stdout";
select(STDERR); $| = 1; # make unbuffered select(STDOUT); $| = 1; # make unbuffered
print STDOUT "stdout 1\n"; # this works for print STDERR "stderr 1\n"; # subprocesses too
close(STDOUT); close(STDERR);
open(STDOUT, ">&OLDOUT"); open(STDERR, ">&OLDERR");
print STDOUT "stdout 2\n"; print STDERR "stderr 2\n";
If you specify
open(FILEHANDLE, "<&=$fd")
If you open a pipe on the command
open(FOO, "|tr '[a-z]' '[A-Z]'"); open(FOO, "|-") || exec 'tr', '[a-z]', '[A-Z]';
open(FOO, "cat -n '$file'|"); open(FOO, "-|") || exec 'cat', '-n', $file; See Safe Pipe Opens for more examples of this.
NOTE: On any operation that may do a fork, any
unflushed buffers remain unflushed in both processes, which means you may
need to set
Closing any piped filehandle causes the parent process to wait for the
child to finish, and returns the status value in The filename passed to open will have leading and trailing whitespace deleted, and the normal redirection characters honored. This property, known as ``magic open'', can often be used to good effect. A user could specify a filename of "rsh cat file |", or you could change certain filenames as needed:
$filename =~ s/(.*\.gz)\s*$/gzip -dc < $1|/; open(FH, $filename) or die "Can't open $filename: $!"; However, to open a file with arbitrary weird characters in it, it's necessary to protect any leading and trailing whitespace:
$file =~ s#^(\s)#./$1#; open(FOO, "< $file\0");
If you want a ``real''
C open() (see open(2) on your system), then you should use the
use IO::Handle; sysopen(HANDLE, $path, O_RDWR|O_CREAT|O_EXCL) or die "sysopen $path: $!"; $oldfh = select(HANDLE); $| = 1; select($oldfh); print HANDLE "stuff $$\n"); seek(HANDLE, 0, 0); print "File contains: ", <HANDLE>;
Using the constructor from the
use IO::File; #... sub read_myfile_munged { my $ALL = shift; my $handle = new IO::File; open($handle, "myfile") or die "myfile: $!"; $first = <$handle> or return (); # Automatically closed here. mung $first or die "mung failed"; # Or here. return $first, <$handle> if $ALL; # Or here. $first; # Or here. } See seek() for some details about mixing reading and writing. |
|