unpackTEMPLATE,EXPR
This function does the reverse of pack: it
takes a string (EXPR) representing a data structure
and expands it out into a list value, returning the list value. (In a scalar
context, it can be used to unpack a single value.) The
TEMPLATE has much the same format as in the pack function - it specifies the order and
type of the values to be unpacked. (See pack for a more detailed description of
TEMPLATE.)
Here's a subroutine that does (some of) substr, only slower:
sub substr {
my($what, $where, $howmuch) = @_;
if ($where < 0) {
$where = -$where;
return unpack "\@* X$where a$howmuch", $what;
}
else {
return unpack "x$where a$howmuch", $what;
}
}and then there's:
sub signed_ord { unpack "c", shift }Here's a complete uudecode program:
#!/usr/bin/perl
$_ = <> until ($mode,$file) = /^begin\s*(\d*)\s*(\S*)/;
open(OUT,"> $file") if $file ne "";
while (<>) {
last if /^end/;
next if /[a-z]/;
next unless int((((ord() - 32) & 077) + 2) / 3) ==
int(length() / 4);
print OUT unpack "u", $_;
}
chmod oct $mode, $file;In addition, you may prefix a field with
%number to
indicate that you want it to return a
number-bit
checksum of the items
instead of the items themselves.
Default is a 16-bit checksum. For example, the following computes the
same number as the System V sum program:
undef $/;
$checksum = unpack ("%32C*", <>) % 32767;The following efficiently counts the number of set bits in a bit vector:
$setbits = unpack "%32b*", $selectmask;
Here's a simple MIME decoder:
while (<>) {
tr#A-Za-z0-9+/##cd; # remove non-base64 chars
tr#A-Za-z0-9+/# -_#; # convert to uuencoded format
$len = pack("c", 32 + 0.75*length); # compute length byte
print unpack("u", $len . $_); # uudecode and print
}