41.5.1. Scalars
When
you want to store single values, like any of those given in the
previous paragraph, you will use a scalar
variable. Scalars are labeled with a $ followed by
a letter and any sequence of letters, numbers, and underscores.
Scalars defined at the top of scripts
are often used as constants. You may need to tweak
some of them, particularly those containing filesystem paths, to get
third-party scripts to run on your system.
Of course, values can be compared to each other or added together.
Perl has relational operators that treat
values as numbers and other relational operators that treat values as
strings. Although Perl has different operators for numbers and
strings, Perl makes scalar values do the right thing most of the
time. For example, you want to create a series of filenames like
mail_num. The following code does this.
foreach my $num (1..10) {
print "mail_" . $num . "\n";
}
Even though $num is a number, the string
concatenation operator is able to use it as a string. Table 40-2 shows string operators, and Table 41-3 shows the numerical ones. See the
perlop manpage for the full story.
Table 41-2. String operators
Operator
|
Example
|
Description
|
.
|
$saluation . " Jones"
|
String concatenation
|
eq
|
$foo eq $bar
|
String equality test
|
ne
|
$bar ne $baz
|
String inequality test
|
gt
|
$name gt "Bob"
|
True if left string comes after right in ASCII
|
lt
|
$name lt "Xavier"
|
True if left string comes before right in ASCII
|
cmp
|
$name cmp "Wilson"
|
Return -1 if left operand ASCII-sorts before the right; 0 if right
and left are equal; 1 if right sorts before left
|
lc
|
lc "Bob"
|
Return an all-lowercase copy of the given string
|
uc
|
uc "lorrie"
|
Return an all-uppercase copy of the given string
|
Operator
|
Example
|
Description
|
+
|
$a + 1
|
Numerical addition
|
-
|
$c - 2
|
Numerical subtraction
|
*
|
3 * $b
|
Numerical multiplication
|
/
|
4/$non_zero
|
Numerical division
|
++
|
$a++
|
Autoincrement; adds one to a number
|
==
|
$a == $b
|
Numeric equality test
|
!=
|
$p != $q
|
Numeric inequality test
|
<
|
$diff < 32
|
Numeric less-than test
|
>
|
$sum > 64
|
Numeric greater-than test
|
<=>
|
$sum <=> 64
|
Return -1 if left is numerically less than right; 0 if left equals
right; 1 if right is less than left
|
<=
|
$sum <= 64
|
True if left operand is numerically less than or equal to right
|
>=
|
$sum >= 64
|
True if left is numerally greater than or equal to right
|
You may have notice that some of the operators in the previous tables
were described as returning true or false values. A true value in Perl
is any value that isn't false, and there are only 4
kinds of false values in Perl:
Like many other languages, Perl supports Boolean operators (see Table 41-3) that return true or false values. Typically,
you encounter these in if statements like the
following:
if ($temp < 30 && $is_rainy) {
print "I'm telecommuting today\n";
}
Another common use of Boolean operators is to short-circuit two
expressions. This is a way to prevent the right operand from
executing unless the left operand returns a desired truth value.
Consider the very ordinary case of opening a filehandle for reading.
A common idiom to do this is:
open (FH, "filename") || die "Can't open file";
This short-cut operation depends on the open
function returning a true value if it can open the requested file.
Only if it cannot is the right side of the ||
operator executed (die prints whatever message you
provide and halts the program).
Table 41-4. Boolean operators
Operator
|
Example
|
Description
|
&&
|
$a && $b
|
True if both $a and $b are true
|
||
|
$a || $b
|
True if either $a or $b is true
|
!
|
!$a
|
True if $a is false
|
and
|
$a and $b
|
Same as &&, but with a lower precedence
|
or
|
$a or $b
|
Same as ||, but with a lower precedence
|
not
|
not $a
|
Same as !, but with a lower precedence
|
Looking at Table 41-4,
you will notice that there appear to be redundant operators. The
operators that are English words have a lower precedence that the
symbolic ones. Precedence is simply the order in which Perl executes
expressions. You are
probably familiar with precedence rules from mathematics:
1 + 2 * 3 + 4 = 11
(1 + 2) * (3 + 4) = 21
Similarly, Perl's operators have precedence as well,
as shown in Example 41-2.
Example 41-2. Precedence
lc $a || "BB" # like (lc $a) || ("BB")
lc ($a || "BB")
Because || has a lower precedence that the
lc operator, the first line of Example 41-2 is a Boolean test between two expressions. In
the second line, the Boolean || operator is used
to create a default argument to lc should
$a be a false value.
Because Perl
doesn't require parentheses around built-in
operators and functions, you will often see code like:
open FH, "> " . "filename" or die "Can't open file";
print FH "[info]: disk write error\n";
Precedence ambiguities can be resolved by using parentheses where
doubt occurs.
Although Perl has many special variables, the one
you'll encounter most is
$_. Many operators and functions, such as
lc and
print,
will operate on $_ in the absence of an explicit
parameter, as in Example 41-3.
Example 41-3. Simple echo loop
while(<>){
print
}
In this example, every line read from standard input with the
<> operator is available inside the
while (Section 41.7)
loop through $_. The print
function, in the absence of an explicit argument, echoes the value of
$_. Note that $_ can be
assigned to (e.g., $_ = "Hello,
Perl") just like any other
scalar.
41.5.3. Hashes
Associative arrays, or
hashes, are
a collection of scalar values that are arranged in
key-value pairs. Instead of using
integers to retrieve values in a hash, strings are used. Hashes begin
with %. Example 41-5 shows a hash
variable in action.
Example 41-5. Using hashes
my %birthdays = (
'mom' => 'JUN 14',
'archie' => 'JUN 12',
'jay' => 'JUL 11',
);
print "Archie's birthday is: ", $birthdays{'archie'}, "\n";
$birthday{'joe'} = 'DEC 12';
print "My birthday is: ", $birthdays{'joe'}, "\n";
Hashes are a funny kind of list. When initializing a hash with
values, it is common to arrange the list in key-value pairs. The
strange-looking => operator is often called a
"fat
comma" because these two lines of Perl do the same
thing:
%birthdays = ( 'jay' => 'JUL 11' );
%birthdays = ( 'jay', 'JUL 11');
Use the fat comma when initializing hashes since it conveys the
association between the values better. As an added bonus, the fat
comma makes unquoted barewords on its left into quoted strings.
Example 41-6 shows some quoting styles for hash keys.
Example 41-6. Various quoting styles for hash keys
my %baz = ( foo => 1,
'bar', 2,
'boz' => 3);
Unlike arrays, hashes use strings to
index
into the list. So to retrieve the birthday of
"jay", put the key inside curly
braces, like this:
print "Jay's birthday is: ", $birthday{'jay'}, "\n";
Because Perl assumes that barewords used as a key when retrieving a
hash value are autoquoted, you may omit quotes between the curly
braces (e.g., $birthday{jay}). Like arrays, hashes
will grow as you need them to. Whenever you need to model a set or
record the number of event occurrences, hashes are the variable to
use.
Like arrays, you will often need to
iterate over the set of key-value pairs
in a hash. Two common techniques for doing this are shown in Example 41-7. Table 41-6 lists common
Perl hash functions.
Example 41-7. Interating over a hash
my %example = (foo => 1, bar => 2, baz => 3);
while (my ($key, $value) = %example) {
print "$key has a value of $value\n";
}
foreach my $key (keys %example) {
print "$key has a value of $example{$key}\n";
}
Table 41-6. Common Perl hash functions
Name
|
Example
|
Description
|
delete
|
delete
$hash{{key"}
|
Delete the key-value pair from hash that is indexed on
key
|
each
|
($key, $value) = each %hash
|
Return the next key-value pair in hash; the pairs
aren't usefully ordered
|
exists
|
print "key found" if exists
$hash{"key"}
|
Return true if hash has key, even if that
key's value if undefined
|
keys
|
@keys = keys %hash
|
Return the list of keys in the hash; not ordered
|
values
|
@values = values %hash
|
Return the list of values in the hash; values will be in the same
order as keys fetched by keys %hash
|