Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Why does !1 give me nothing in Perl?

Tags:

boolean

perl

This is strange. The following:

$sum = !0;
print $sum;

prints out 1 as you would expect. But this

$sum = !1;
print $sum;

prints out nothing. Why?

like image 269
Vlad the Impala Avatar asked Jul 16 '09 01:07

Vlad the Impala


People also ask

What $_ means in Perl?

There is a strange scalar variable called $_ in Perl, which is the default variable, or in other words the topic. In Perl, several functions and operators use this variable as a default, in case no parameter is explicitly used.

What does ~= mean in Perl?

=~ is the Perl binding operator and can be used to determine if a regular expression match occurred (true or false) $sentence = "The river flows slowly."; if ($sentence =~ /river/) { print "Matched river.\n"; } else { print "Did not match river.\n"; } Follow this answer to receive notifications.

Is 0 true or false in Perl?

False Values: Empty string or string contains single digit 0 or undef value and zero are considered as the false values in perl.

What is $$ in Perl?

$$ The process number of the perl running this script. (Mnemonic: same as shells.) $? The status returned by the last pipe close, backtick (\`\`) command or system operator.


3 Answers

Be careful: what you've written isn't doing what you think it's doing. Remember, perl has no real boolean datatype. It's got scalars, hashes, lists, and references. The way it handles true/false values, then, is contextual. Everything evaluates to "true" in perl except for undefined variables, the empty list, the empty string, and the number 0.

What your code is doing, then, is taking the inverse of a value that evaluates to "false", which can be anything which is not in the list above. By convention and for simplicity's sake, perl returns 1 (though you should not rely on that; it could very well return a list containing a series of random numbers, because that will evaluate to "true" as well.)

A similar thing happens when you ask for the inverse of a value that evaluates to "true." What's actually being printed out is not "nothing," it's the empty string (''), which, as I mentioned, evaluates to "false" in boolean expressions. You can check this:

print "This evaluates to false\n" if( (!1) eq '');

If you're asking for why perl spits out the empty string instead of one of the other "false" values, well, it's probably because perl is made to handle strings and that's a perfectly reasonable string to hand back.

like image 158
Omar Zakaria Avatar answered Sep 25 '22 11:09

Omar Zakaria


The operators that only return a boolean result will always return 1 for true and a special false value that's "" in string contexts but 0 in numeric contexts.

like image 9
ysth Avatar answered Sep 24 '22 11:09

ysth


Here's an addendum to the other great answers you've already gotten.

Not's Not Not

Consider the following code that tests each of Perl's 'not' operators:

#!/usr/bin/perl

use strict;
use warnings;

for( '!1', 'not 1', '~0' ) {
    my $value = eval;
    my $zero_plus = 0 + $value;

    print join "\n", 
        "\nExpression: $_",
        "Value:     '$value'",
        "Defined:   " . defined $value,
        "Length:    " . length($value),
        "Plus:      " . +$value,
        "Plus Zero: '$zero_plus'",
        '';
}

print "\nTest addition for a literal null string:  ";
print 0+'', "\n";

use Scalar::Util qw(dualvar);

{   # Test a dualvar 
    my $value = dualvar 0, '';
    my $zero_plus = 0+$value;

    print join "\n", 
        "\nExpression: dualvar",
        "Value:     '$value'",
        "Defined:   " . defined $value,
        "Length:    " . length($value),
        "Plus:      " . +$value,
        "Plus Zero: '$zero_plus'",
        '';

}

Executing it results in the following. Notice the warning message:

Argument "" isn't numeric in addition (+) at test.pl line 21.

Expression: !1
Value:     ''
Defined:   1
Length:    0
Plus:
Plus Zero: '0'

Expression: not 1
Value:     ''
Defined:   1
Length:    0
Plus:
Plus Zero: '0'

Expression: ~0
Value:     '4294967295'
Defined:   1
Length:    10
Plus:      4294967295
Plus Zero: '4294967295'

Test addition for a literal null string:  0

Expression: dualvar
Value:     ''
Defined:   1
Length:    0
Plus:
Plus Zero: '0'

From this we learn several things.

The first two items are not all that exciting:

  • !1 and not 1 behave in basically the same way.
  • Unsurpisingly, ~1 is different (it's the bitwise not).

Now, the interesting item:

  • While we do get a warning for line 21 (0+''), there is no warning generated when we add 0+!1.

It Takes Two to Tangle

Something fishy is happening, and that fishiness has to do with special scalar contexts in Perl. In this case, the distinction between numeric and string contexts. And the ability to create a variable that has different values in each context, aka a dual variable.

It looks like !1 returns a dual variable that returns 0 in numeric context and the null string in string context.

The dualvar test at the end shows that a homemade dualvar works the same way as !1.

But It's A Good Thing

Like many Perl features, dual variables seem at first to defy expectations, and can be confusing. However, like those other features, used appropriately they make life much easier.

As far as I know, a dualvar of 0 and '' is the only defined value that will return false in all scalar contexts. So it is a very sensible return value for !1. One could argue that undef is a good false result, but then an uninitialized variable is not distinguishable from a false value. Also, attempts to print or add the results of booleans would then be plagued with unnecessary warnings.

Another famous dualvar is $! or $OS_ERROR if you use English. In numeric form, you get the error code, in string form, the error code is translated for you.

It's Not Nothing, It's Empty, But It's Nought

So in summary, you aren't getting nothing, you aren't getting an empty string, and you aren't getting zero.

You are getting a variable that is both an empty string and 0 at the same time.

like image 6
daotoad Avatar answered Sep 28 '22 11:09

daotoad