NAME

Inline::SLang::Struct - Support for structures

SYNOPSIS

use Inline 'SLang';

# you can send arrays to S-Lang
# - note we use them like a hash reference
my $s1 = Struct_Type->new( ["xx","aa","one"] );
$$s1{xx} = 14; $$s1{aa} = "foo"; $$s1{one} = [1,2];
print_in_slang( $s1 );

# and get them back from S-Lang
my $s2 = get_from_slang();
print "The struct has a type of $s2\n";
while ( my ( $k, $v ) = each %$s2 ) {
  print "  field $k has type ",
    (ref($v) ? ref($v) : "perl scalar"),
    " and value $v\n";
}

__END__
__SLang__

define print_in_slang (st) {
  vmessage( "S-Lang has been sent a %S structure", typeof(st) );
  foreach ( get_struct_field_names(st) ) {
    variable f = ();
    vmessage( " and field %s = %S", f, get_struct_field(st,f) );
  }
}
typedef struct { key1, wowza } FooStruct;
define get_from_slang() {
  variable x = @FooStruct;
  x.key1 = "this is key 1";
  x.wowza = 4+3i;
  return x;
}

The output of this code - which can be found in the source-code distribution as examples/structs.pl - is:

S-Lang has been sent a Struct_Type structure
 and field xx = 14
 and field aa = foo
 and field one = Integer_Type[2]
The struct has a type of FooStruct
  field key1 has type perl scalar and value this is key 1
  field wowza has type Math::Complex and value 4+3i

DESCRIPTION

S-Lang's structures can be thought of as "restricted" associative arrays, in that the keys are fixed (no adding or deleting) and when you loop through them the order is set by the orer they are stored in the structure. So, we therefore convert S-Lang structures into Struct_Type objects which behave like hash references (with some additional methods).

To support "named" structures - i.e. those created using S-Lang's typedef struct { ... } SomeName; syntax, we create sub-classes of the Struct_Type class with names that match that of the structure. In this particular case that would be SomeName. Note that this only works for those typedef statements executed when the code is loaded; so if you call S-Lang's eval() function with a typedef statement then that structure name will not be available as a Perl class.

Note that a bare hash reference is not converted into a S-Lang structure since this is used to represent an associative array (see Inline::SLang::Assoc).

The Struct_Type Class

Once created, the Perl Struct_Type class behaves like a hash reference except for the following:

  • Keys can not be created once the object has been created.

  • Keys can not be destroyed (although their contents can be set to undef).

  • When iterating through the keys or values, the order is fixed to match that of the field names in the structure.

The Struct_Type class inherit the default methods of all the Inline::SLang objects, namely:

  • typeof()

    This returns Struct_Type as a DataType_Type object. For "named" structures it returns the data type of the structure.

  • stringify()

    This returns the string "Struct_Type". For "named" structures it returns the name of the structure.

  • is_struct_type()

    This returns 1.

As with the other object classes these methods can only be called using the $object->method( args ) syntax. There are also a number of additional methods for this class:

  • new( field-names )

    The constructor takes an array reference which contains the field names and returns an object in which the values of these fields are set to undef. For "named" structs the field names array is not needed since the fields are fixed for these structures.

    The return value can be treated as a hash reference, or you can use some of the other methods described below to manipulate the contents of the array.

    my $struct = Struct_Type->new( ["x","a","one"] );
    $$struct{one} = 2;
    $$struct{x} = "hello";
    $$struct{a} = "hi there";
    while ( my ($k,$v) = each %$struct ) {
      print "  key $k has a value of $v\n";
    }

    The order of keys returned by the Perl hash iterators - such as each, keys, and values - is set to the order of the fields in the structure. For this example it would be x, a, and one.

  • get_field_names()

    Returns, as an array reference, a list of the fields of the structure, in the order they have in the structure. It is essentially S-Lang's struct_get_field_names() routine although it is implemented using Perl's keys routine.

    my $struct = Struct_Type->new( ["x","a","one"] );
    foreach my $f ( $struct->get_field_names ) {
      print "  field = $f\n";
    }

    There is no advantage to using this method rather than Perl's keys %$struct.

  • get_field( $field )

    Returns the value of the given field in the structure. It is essentially S-Lang's struct_get_field() routine.

    my $struct = Struct_Type->new( ["x","a","one"] );
    $$struct{one} = 2;
    $$struct{x} = "hello";
    $$struct{a} = "hi there";
    print "Field x has a value of ", $struct->get_field("x"), "\n";

    There is no advantage to using this method rather than Perl's $$struct{$key}.

  • set_field( $field, $value )

    Sets the value of the given field in the structure. It is essentially S-Lang's struct_set_field() routine.

    my $struct = Struct_Type->new( ["x","a","one"] );
    $struct->set_field( "one", 2 );
    $struct->set_field( "x", "hello" );
    $struct->set_field( a => "hi there" );

    There is no advantage to using this method rather than Perl's $$struct{$key}.

S-Lang's stuct_set_fields() routine is currently not provided by the object but it's not hard to add (or to emulate).

Examples

Another simple example of using structures in Perl (this can be found in examples/types_struct.pl) is shown below:

use Inline 'SLang' => <<'EOS';
variable runtime = time();
typedef struct { x, time } xTime_Struct;
define ret1(x) {
  variable y = struct { x, time };
  y.x    = x;
  y.time = runtime;
  return y;
}
define ret2(x) {
  variable y = @xTime_Struct;
  y.x    = x;
  y.time = runtime;
  return y;
}
EOS

# first with a normal structure
my $s1 = ret1( "struct example" );
print  "ret1() returned a $s1\n";
printf "Is it a structure? [%d]\n", $s1->is_struct_type;
printf "With keys/fields [ %s ]\n",
  join( ", ", keys(%$s1) );
print  " s.x    = $$s1{x}\n";
print  " s.time = $$s1{time}\n";

# and then with a "named" structure
my $s2 = ret2( "named struct example" );
print  "ret2() returned a $s2\n";
printf "Is it a structure? [%d]\n", $s2->is_struct_type;
printf "With keys/fields [ %s ]\n",
  join( ", ", keys(%$s2) );
print  " s.x    = $$s2{x}\n";
print  " s.time = $$s2{time}\n";

The output of the code is something like (depending on whether you keep sociable hours or not):

ret1() returned a Struct_Type
Is it a structure? [1]
With keys/fields [ x, time ]
 s.x    = struct example
 s.time = Sat May  3 03:46:01 2003
ret2() returned a xTime_Struct
Is it a structure? [1]
With keys/fields [ x, time ]
 s.x    = named struct example
 s.time = Sat May  3 03:46:01 2003

An example of creating a "named" structure from Perl (available as examples/named_struct.pl in the source doe) is:

use Inline 'SLang' => <<'EOS';
typedef struct { x, foo } My_Struct;
define is_okay(x) {
  if ( typeof(x) != My_Struct ) {
    vmessage("You sent me a %S", typeof(x));
    return;
  }
  vmessage( "My_Struct field x   = %S", x.x );
  vmessage( "My_Struct field foo = %S", x.foo );
}
EOS

my $s = My_Struct->new();
$$s{x}   = 1;
$$s{foo} = "foo foo";
is_okay( $s );

which produces:

My_Struct field x   = 1
My_Struct field foo = foo foo

SEE ALSO

Inline::SLang::Assoc, Inline::SLang