http://www.perlmonks.org?node_id=11160559

colintu has asked for the wisdom of the Perl Monks concerning the following question:

I have an Entry widget where the textvariable is initialised as a numeric before the widget is created. The widget and validate routine looks like:

my @c_speed = (5, 8, 12, 8); my $cs_e1 = $seg1_f->Entry(-textvariable => \$c_speed[0], -width => +4, -validate => 'all', -vcmd => \&validate_speed, -font => $med_font)->grid( -row => 0, -column => 1, -sticky => "w"); sub validate_speed{ my $val = shift; $val ||= 0; #get alphas and punctuation out if( $val !~ /^\d+$/ ){ return 0 } if (($val >= 0) and ($val <= 20)) {return 1} else{ return 0 } }

To me this looks like $val is being treated as a string (if( $val !~ /^\d+$/ )) and then on the next line as a numeric variable ($val >= 0) etc. The code does what I want/expect but how? Is this just 'magic behind the curtain'?

Replies are listed 'Best First'.
Re: Tk Entry widget confusion
by hv (Prior) on Jul 12, 2024 at 11:19 UTC

    This is probably the primary magic of Perl: values are silently converted to the type required for the operator acting on them as far as possible. Behind the curtain, it caches those conversions and remembers whether they are exact or lossy.

    This also allows for fun tricks, such as "dualvars" that have a different cached value in the string slot and the integer slot - that's used, for example, to allow $! to reveal both the error message and the error number:

    % perl -wE 'stat("nonexistent"); $e = $!; say $e; say $e+0' No such file or directory 2

    You can see what's going on in gory detail with a module such as Devel::Peek:

    % perl -MDevel::Peek -we ' $val = ""; Dump($val); $val ||= 0; Dump($val); $val =~ /^\d+$/; Dump($val); ' 2>&1 | grep V SV = PV(0x561cd5e52da0) at 0x561cd5e7a028 PV = 0x561cd5e9b150 ""\0 SV = PVIV(0x561cd5e74e60) at 0x561cd5e7a028 IV = 0 PV = 0 SV = PVIV(0x561cd5e74e60) at 0x561cd5e7a028 IV = 0 PV = 0x561cd5e7b6a0 "0"\0

    $val is a scalar value ("SV") as opposed to an array value ("AV"), hash value ("HV"), or something more exotic. The SV has slots for a string ("pointer value" = "PV"), an integer ("IV") and a (floating-point) numeric value ("NV").

    In the example above, we start off with a plain string (PV = ""). Then we replace it with an integer 0 (IV = 0, PV is cleared out); then we use that in a string context so it generates and caches the stringified version (PV = "0").

    The hex values in there are memory addresses. You don't normally care about them, but it can be useful to see whether two memory addresses are the same or not.

    I've grepped out most of the gore here; that includes the reference count (which Perl uses to know when a value should be freed); the length of the string value (CUR) and how much memory is allocated for it (LEN, confusingly); flags that say which of the value slots are valid (IOK, POK, NOK) and partially valid (pIOK, pPOK, pNOK); and a flag (IsCOW) and associated reference count if the string is sharable, as a memory saving.

    Almost all of the time you don't need to know any of this, but having at least some clue what's going on behind the curtain can be very helpful to make Perl seem more predictable.

      Thanks, that does explain what I'm seeing.
Re: Tk Entry widget confusion
by hippo (Archbishop) on Jul 12, 2024 at 10:54 UTC
    The code does what I want/expect but how? Is this just 'magic behind the curtain'?

    Welcome to Perl! This is how it rolls.

    Variables in Perl (including arrays, hashes, subroutines, etc.) are not strongly typed. This means that most often you do not need to worry about what is a number and what is a string, even less what is an integer and what is a float, etc.


    🦛

      I always have it in my mind that 'not strongly typed' means you can use them for anything. Perl takes that to a higher level it seems and uses it to mean 'automatic type and content conversion in real-time'!