11.3. Fatal Errors with die
Let's
step aside for a moment. We need some stuff
that isn't directly related to (or limited to) filehandles, but
is more about getting out of a program earlier than normal.
When a fatal error happens inside Perl (for example, if you divide by
zero, use an invalid regular expression, or call a subroutine that
hasn't been declared) your program stops with an error message
telling why.[253]
But this functionality is available to us with the
die function, so we can make our own fatal
errors.
The die function prints out the
message you give it (to the standard
error stream, where such messages should go) and makes sure that your
program exits with a nonzero exit status.
You may not have known it, but every program that runs on Unix (and
many other modern operating systems) has an exit status, telling whether it was
successful or not. Programs that run other programs (like the
make utility program) look at that exit status
to see that everything is running correctly. The exit status is just
a single byte, so it can't say much; traditionally, it is zero
for success and a nonzero value for failure. Perhaps one means a
syntax error in the command arguments, while two means that something
went wrong during processing and three means the configuration file
couldn't be found; the details differ from one command to the
next. But zero always means that everything worked. When the exit
status shows failure, a
program like make knows
not to go on to the next step.
So we could rewrite the previous example, perhaps something like this:
unless (open LOG, ">>logfile") {
die "Cannot create logfile: $!";
}
If the open fails, die will
terminate the program and tell us that it cannot create the logfile.
But what's that $! in the message?
That's the human-readable complaint from the system. In
general, when the system refuses to do something we've
requested (like opening a file), it will give us a reason (perhaps
"permission denied" or "file not found," in
this case). This is the string that you may have obtained with
perror in C or a similar language. This
human-readable complaint message will be available in Perl's
special variable $!.[254] It's a good idea to include
$! in the message when it could help the
user to figure out what he or she did wrong. But if you use
die to indicate an error that is not the failure
of a system request, don't include $!, since
it will generally hold an unrelated message left over from something
Perl did internally. It will hold a useful value only immediately
after a failed system request. A successful
request won't leave anything useful there.
There's one more thing that die will do
for you: it will automatically append the Perl
program name and line number[255] to
the end of the message, so you can easily identify which
die in your program is responsible for the
untimely exit. The error message from the previous code might look
like this, if $! contained the message
permission denied:
Cannot create logfile: permission denied at your_program line 1234.
That's pretty helpful -- in fact, we always seem to want
more information in our error messages than we put in the first time
around. If you don't want the line number and file revealed,
make sure that the dying words have a newline on the end. That is,
another way you could use die is in a line like
this, with a trailing
newline:
die "Not enough arguments\n" if @ARGV < 2;
If there aren't at least two command-line arguments, that
program will say so and quit. It won't include the program name
and line number, since the line number is of no use to the user; this
is the user's error, after all. As a rule of thumb, put the
newline on messages that indicate a usage error and leave it off when
it the error might be something you want to track down during
debugging.[256]
When opening a file fails, though, there's an easier and more
common way instead of the unless block:
open LOG, ">>logfile"
or die "Cannot create logfile: $!";
This uses the low-precedence short-circuit or operator that we saw
in Chapter 10, "More Control Structures". If the open
succeeds, it returns true, and the or is done. If
the open fails, it returns false, and the
short-circuit or goes on to the right side and
dies with the message. You can read this as if it were English:
"Open this file, or die!" It may not be the battle cry
that will win a war, but it's a good way to write code.
You should always check the return value of
open, since the rest of the program is relying
upon its success. That's why we say that this is really the
only way to write open -- with or
die after it.[257] Until you're ready to be
extra tricky, you should simply think of this as the syntax for
open. Typing or die and a
message takes only a moment when you're writing the program,
but it can save hours, or possibly days of debugging time when
something goes wrong.
 |  |  | 11.2. Opening a Filehandle |  | 11.4. Using Filehandles |
Copyright © 2002 O'Reilly & Associates. All rights reserved.
|