Функция defined() в Perl

В языке программирования Perl есть встроенная функция defined().

Чаще всего эта функция используется для того чтобы отличить значение undef от любого другого значения. Но еще с помощью defined() можно узнать определена ли функция.

Проверка на undef

Основное примение функции defined() это проверка значения на undef.

Если передать функции defined() значение undef, то функция вернет ложь. Функция вернет истину в том случае если значение которое ей передано является чем угодно, но не undef. Вот пример кода:

▶ Run
#!/usr/bin/perl

use Data::Dumper;

print Dumper defined(8);
print Dumper defined(0);
print Dumper defined('');
print Dumper defined(undef);

Результат работы этого кода:

$VAR1 = 1;
$VAR1 = 1;
$VAR1 = 1;
$VAR1 = '';

Проверка на то что функция определена

Чаще всего функция defined() используется для того чтобы отличить undef от другого значения. Но кроме этого с помощью defined() можно узнать определена ли функция. Вот пример кода:

▶ Run
#!/usr/bin/perl

use strict;
use warnings;

use Data::Dumper;

print Dumper defined(&check);
print Dumper defined(&Data::Dumper::Dumper);

Функция check() не определена, так что defined() вернет ложь, А вот функция Dumper есть в пакете Data::Dumper и defined() вернет истину. Вот что выведет эта программа на экран:

$VAR1 = '';
$VAR1 = 1;

Аргументы

В том случае если функции defined() не переданы никакие аргументы, то функция работает с дефолтной переменной $_:

Если передать функции defined() более чем один аргумент, то будет ошибка и выполнение программы будет остановлено:

Too many arguments for defined operator at script.pl line 3, near "3)"
Execution of script.pl aborted due to compilation errors.

Возвращаемое значение

Результат работы функции defined() — это всегда булево значение. Истина или ложь.

Использование с массивом и хешом

Очень давно функция defined() работала особым образом если ей передать массив или хеш в качестве аргумента. Но уже в Perl 5.8 такое использование выдавало предупреждение, а начиная с 5.22 такое использование стало выдавать ошибку.

Вот пример кода в котором в defined() передается массив:

▶ Run
#!/usr/bin/perl

use strict;
use warnings;

use Data::Dumper;

my @arr;

print Dumper defined(@arr);

Результат работы этого кода в perl версии 5.8:

defined(@array) is deprecated at script.pl line 10.
    (Maybe you should just omit the defined()?)
    $VAR1 = '';

И результат работы этого же когда в perl 5.30:

Can't use 'defined(@array)' (Maybe you should just omit the defined()?) at script.pl line 10.

И такой же код, но про хеш:

▶ Run
#!/usr/bin/perl

use strict;
use warnings;

use Data::Dumper;

my %h;

print Dumper defined(%h);

Результаты работы на perl 5.8 и 5.30:

defined(%hash) is deprecated at script.pl line 10.
    (Maybe you should just omit the defined()?)
    $VAR1 = '';
Can't use 'defined(%hash)' (Maybe you should just omit the defined()?) at script.pl line 10.

Официальная документация

Вот вывод команды perldoc -f defined:

       defined EXPR
       defined Returns a Boolean value telling whether EXPR has a value other
               than the undefined value "undef".  If EXPR is not present, $_
               is checked.

               Many operations return "undef" to indicate failure, end of
               file, system error, uninitialized variable, and other
               exceptional conditions.  This function allows you to
               distinguish "undef" from other values.  (A simple Boolean test
               will not distinguish among "undef", zero, the empty string, and
               "0", which are all equally false.)  Note that since "undef" is
               a valid scalar, its presence doesn't necessarily indicate an
               exceptional condition: "pop" returns "undef" when its argument
               is an empty array, or when the element to return happens to be
               "undef".

               You may also use "defined(&func)" to check whether subroutine
               &func has ever been defined.  The return value is unaffected by
               any forward declarations of &func.  A subroutine that is not
               defined may still be callable: its package may have an
               "AUTOLOAD" method that makes it spring into existence the first
               time that it is called; see perlsub.

               Use of "defined" on aggregates (hashes and arrays) is
               deprecated.  It used to report whether memory for that
               aggregate had ever been allocated.  This behavior may disappear
               in future versions of Perl.  You should instead use a simple
               test for size:

                   if (@an_array) { print "has array elements\n" }
                   if (%a_hash)   { print "has hash members\n"   }

               When used on a hash element, it tells you whether the value is
               defined, not whether the key exists in the hash.  Use "exists"
               for the latter purpose.

               Examples:

                   print if defined $switch{D};
                   print "$val\n" while defined($val = pop(@ary));
                   die "Can't readlink $sym: $!"
                       unless defined($value = readlink $sym);
                   sub foo { defined &$bar ? &$bar(@_) : die "No bar"; }
                   $debugging = 0 unless defined $debugging;

               Note:  Many folks tend to overuse "defined" and are then
               surprised to discover that the number 0 and "" (the zero-length
               string) are, in fact, defined values.  For example, if you say

                   "ab" =~ /a(.*)b/;

               The pattern match succeeds and $1 is defined, although it
               matched "nothing".  It didn't really fail to match anything.
               Rather, it matched something that happened to be zero
               characters long.  This is all very above-board and honest.
               When a function returns an undefined value, it's an admission
               that it couldn't give you an honest answer.  So you should use
               "defined" only when questioning the integrity of what you're
               trying to do.  At other times, a simple comparison to 0 or ""
               is what you want.

               See also "undef", "exists", "ref".

Связанные темы

Другие статьи