Getting started with OOP for User Language programmers
Background
So, you're a User Language programmer and you're thinking about learning object-oriented programming:
- Rumor has it you can be a more effective programmer if you use object-oriented techniques.
- You're tired of feeling inferior to object-oriented programmers because they speak a language that you don't understand but sure as heck sounds impressive.
- You use Sirius Software products and Sirius has announced that all new User Language functionality will used object-oriented syntax.
- “Object-oriented” looks better on your resume than “User Language”.
- You just want to learn something new.
Unfortunately, most User Language programmers' first experience with object-oriented programming is painful and bewildering. Often it comes in the form of a VB.Net or Java class where the terminology flows freely from day one. Even worse:
- Classes often emphasize how one architects an object-oriented application. While this might be a logical way to build an application, it's a daunting way to learn a language — like trying to learn ballroom dancing before you know how to walk.
- Teachers (and books about object-oriented programming) are often enamored with the more sophisticated aspects of object-oriented programming languages, leaving novices in the dust as they're still struggling to digest the simpler concepts.
- Many object-oriented concepts are inter-related so it often requires plowing ahead without fully understanding the concepts one has already learned.
- There seems no way to put the concepts learned in a Java or VB.Net class to use in User Language. So, one is either forced to try to work on object-oriented programming in one's free time, or the concepts are quickly forgotten.
Fortunately, there is a way you can learn object-oriented programming, and apply the principles on-the-job from day one!
One requirement for this is that one work at a site where Janus SOAP User Language Interface is available or, at least some of the other Sirius “API” products such as Janus Web Server or Janus Sockets.
First, a quick word about terminology. In English-speaking, as opposed to American-speaking countries, objected-oriented is usually called object-orientated. More commonly, everyone just calls object-oriented programming “O-O”. But then, you knew that.
Mixed case code
So, let's get started. First, if you're going to do O-O programming you've got to write your code in mixed case. No, there is no technical reason O-O code in User Language must be in mixed case but:
- It's easy enough to do and, even if you don't use O-O, it makes your code look more modern.
- The industry consensus is that descriptive, often compound words, are better for variable, function, and subroutine names than terse non-descriptive names. For example, %itemNumber is better than %ITMN. While the latter is easier to type, the former is much easier to read. USing %ITEMNUMBER, on the other hand, clearly blunts some of that readability benefit. %itemNumber is written in what's called CamelCase.
- Sirius Software depends on CamelCase to make their function and subroutine names readable.
Fortunately, it's easy to start using mixed case code. Simply start typing your code in mixed case. What can possibly go wrong? If you're system manager has set everything up nicely for you, nothing. But, if not, you might hot a few glitches that are easy enough to fix:
- If you use the 204 editor and type in mixed case code and it gets converted to upper case when you hit enter, you have two options:
- Set *LOWER at command level. This is a bit of a hassle because now Model 204 commands require holding the shift key down, because mixed case Model 204 command don't work.
- Set the SIREDIT user parameter to X'33' (only the X'01' bit is required, but you may as well set some others) before entering the editor. This causes Model 204 to essentially switch to *LOWER mode before entering the editor. Request that your system manager do this for everyone by setting SIREDIT X'33' in CCAIN.
 
- If you get compilation errors when you type in mixed case UL, it means that the compiler is not running in case-independent mode. To fix this, you have three options:
- Have your system manager set the X'01' bit in the system COMPOPT parameter. This must be done in CCAIN and is probably the best/simplest option.
- Change the BEGIN or B statement at the start of the request you're working on to use mixed case, as in Begin, begin, or b.
- If you can't change the start of the program, add the line Sirius Case ToUpper (the case of the words, doesn't matter) to the start of the procedure you're working on.
 
The following is an example of a mixed case User Language program that starts with a mixed case Begin:
begin print 'Hello World!' end
The following is an example of an Included procedure that contains a Sirius Case ToUpper directive:
 sirius case toUpper
 subroutine hello
    print 'Hello World!'
 end subroutine
Note that mixed case UL support is case independent so you can write UL statements in an case, and you can refer to Model 204 variables in any case. The following illustrates a little island of mixed case code in the middle of some upper case code:
 SUBROUTINE FOOBAR(%INPUT IS FLOAT)
    %MSG   IS STRING LEN 32
    %TOTAL IS FLOAT
    ...
    if %input gt %total then
       %msg = 'Input value too big'
    end if
    ...
    PRINT %MSG
 END SUBROUTINE
This illustrates that you don't have to convert an entire procedure (or request) to mixed case to take advantage of mixed case UL, though, obviously, in the long-term, it's a good goal to aim for relatively consistent casing in all your code. In the short term, however, a bit of inconsistency will have to be tolerated to get to the point where most or all UL code is in mixed case. Certainly, any new procedures should be written completely in mixed case.
As this example, illustrates, you don't have to learn anything new to enter UL in mixed case (all statements still work the same way) so there is no excuse not to start.
Object-oriented syntax
The most pernicious difference between O-O languages and procedural languages such as User Language (we'll just call it UL from here on) is the syntax. And the biggest syntactic difference between O-O and UL is how functions and subroutines are invoked. Let's start with functions. All UL programmers know how to invoke a function.
First, functions are called $functions (dollar-functions) or £functions (pound-functions) in the UK. $functions (except some of the Sirius $functions) always return a value so must either be on the right side of an assignment, input to a subroutine or other $function call, or inside some UL expression. The following example, has a $substr in all three contexts:
%x = $substr(%y, 3, 10) call clever($substr(%y, %start, %len)) %z = $substr(%y, %len, 1) + 10
As the above example illustrates, and all UL programmers know, a $function can be followed by the $function arguments (inputs) in parentheses with multiple arguments separated by commas.
O-O functions, on the other hand, use a syntax where a function invocation consists of the thing (object, if you will) that the function is operating on specified before the function name, followed by its arguments inside parentheses. Many $functions have O-O equivalents and $substr is no exception: its O-O equivalent is called substring. The following illustrates the use of the O-O substring function by replacing $substr in the previous example with substring:
%x = %y:substring(3, 10) call clever(%y:substring(%start, %len)) %z = %y:substring(%len, 1) + 10
While this might look strange to a User Language programmer, it uses the most common O-O syntax so can honestly be called O-O programming. So, if you find any $substr call in your system and change it to use substring (moving the first argument before a :substring) you have now done some O-O coding. It's that easy!
To further add to your bona fides as an object oriented programmer, don't call substring a function, but call it a method. In addition, don't call %y just a string, call it a string object. Now you're ready for something more advanced. Say “I applied the substring method to the string object”. Repeat until it feels natural to say it. Congratulations, you're well on your way to becoming an O-O programmer and, maybe even a guru. If you're curious about what you just said:
- Method is just a fancy word that means function or subroutine or any other called code that does something.
- Object is just a fancy word for a thingy.
Now, of course, there are many functions available other than just Substring. The functions that operate on strings are called Intrinsic String Methods. You can find a list of the methods at List of Intrinsic String Methods. There are also functions that operate on numbers. The list of these can be found at List of Intrinsic Float Methods.
Now, while congratulating yourself on your new-found skills, you might have the gnawing feeling inside that you really haven't accomplished a lot, as much as you have learned. So what have you accomplished? Is O-O syntax really better than what you're used to? At first blush, it would seem worse, as the traditional $function call is more like English, where verb (function name) precedes the object (first parameter), as opposed to the O-O syntax, where the order is reversed. For example, one would say “get the substring of %y” not “%y get the substring” so
%x = $substr(%y, 2, 3)
seems more natural than
%x = %y:substring(2, 3)
While this might be true, it's easy enough to get used to the second form — in many languages the object comes before the verb.
In any case, the chief advantage of O-O syntax is that because the input of a function is to the left of the function, one can invoke a function and then pass the result of that function to another function by placing the second function call after the first. Similarly, a third function can be placed to the right of second to process its output. To illustrate, consider the following:
%x = %y:substring(%start, %len):toUpper:unspace
This reads rather nicely, left to right: take %y get a substring, convert to upper case, and remove extra blanks. The traditional version doesn't read nearly as nicely:
%x = $unspace($upcase($substr(%y, %start, %len)))
Since the processing happens in inside out order, presumably one should read this code inside out but, of course, this is very difficult. In addition, because only a colon is required as a separator character, and because no dollar-sign is required to distinguish a function from other entities, the O-O version is shorter, in spite of the fact that the function names are somewhat longer, and so, more meaningful. Perhaps more important, the O-O expression contains fewer “noise” characters. All of these lead to much better readability for the O-O expression. And, just as spaces can be placed around parentheses in a $function invocation to improve readability, spaces can be placed around the colon used to separate the object and function name:
%x = %y : substring( %start, %len ): toUpper :unspace
This example used inconsistent spacing just to illustrate what's allowed, not to suggest that inconsistent spacing is recommended — it's not.
So, to continue the process of becoming an object-oriented User Language programmer, you should familiarize yourself with the list of intrinsic string and float methods and try to use them wherever possible and try to use them in lieu of the $function equivalents. There is no performance penalty for doing so — in some cases O-O functions and $functions share the same underlying code.
Named parameters
The Sirius object-oriented User Language implementation has support for named parameters, parameters that can be specified by name, rather than position. While, strictly speaking, this has nothing to do with O-O programming (in fact, few O-O languages support it -- Java and VB.Net do not) named parameters are used in many Sirius methods, so it is important to understand them. To specify a value for a named parameter, simply specify the name, followed by an equals sign (=), followed by the value. For example, the Right and Left String functions have a Pad named parameter that indicates the pad character to be used if the input string is shorter than the requested length:
%x = %y:right(20, pad='0')
In this case, the name simply makes the code clearer as without the name:
%x = %y:right(20, '0')
it's far less obvious what the second parameter means. In functions with large numbers of parameters, the named parameters can also be very useful for eliminating the need for placeholder commas, and for making the function invocations more readable. String and Float methods tend not to have a lot of parameters so named parameters are not heavily used for them, but they are used here and there, so it's important to understand them.
Stringlists
The object-oriented extensions to User Language are provided by Sirius Software and are only available at sites that can use $lists. Most sites that can use $lists, do use $lists because many Sirius $functions require them as input or outputs, and because they are just generally useful. $lists are essentially objects because an object is a container for information that is accessed via a reference variable, and multiple reference variables can refer to the same object ($list). For example:
%list is float %list2 is float ... %list = $listNew %list2 = %list $listAdd(%list, 'Now is the winter') print $listInf(%list2, 1)
In this example, the Print statement would end up printing “Now is the winter” even though the $listAdd added the line to %list. Both %list and %list2 ;point to the same $list (object) and, so, it is not surprising that what is added via %list can be seen via %list2. It is clear, too, that %list and %list2 must be pointers to $list objects and cannot be the objects, themselves, since a Float value couldn't possibly hold the contents of a $list. An experienced UL programmer might raise an eyebrow at the line that contains the $listAdd in the above example. $listAdd is a function and, so, returns a value. But the $listAdd is not on the right side of an assignment. What gives? Well, the result of many $functions is usually “uninteresting” to the UL code so it seems silly to have to find a variable to which to assign it. So, the Sirius Mods allows certain $functions to be invoked without assigning its result to anything. This causes the $function to behave more like a subroutine as subroutines are called without obtaining a return value from the subroutine. It is this blurring of the distinction between functions and subroutines that leads to the O-O term method which is a convenient way of referring to either.
So, when you were using $lists, you were using object-oriented programming capabilities even though you might not have realized it. However, you were not using object-oriented syntax to access the $list objects and so the benefits of a pure object-oriented facility were not available to you. Now, you might just say “Hold on, didn't you say strings are also objects?&rdquo. Indeed they are. However, they're objects that can't be changed once they're set (so-called immutable objects) — you can't really change a string that's been assigned to a variable, you can only assign a new string to that variable. Most objects are not immutable though, of course, strings and numbers are heavily used, so have importance disproportionate to their number. Nevertheless, the thing that distinguishes O-O languages from non-O-O languages is the presence of changeable (non-immutable) objects so $lists can be viewed as a step toward object-oriented programming.
The pure object-oriented equivalennt of $lists are called Stringlists. And, using a Stringlist is very similar to using a $list. To illustrate, let's take the above example and convert it to using Stringlists:
 %list     is object stringlist
 %list2    is object stringlist
  ...
 %list  = new
 %list2 = %list
 %list:add('Now is the winter')
 print %list2:item(1)
Your first reaction might be an understandable “So what?” Well, let's take a look at what we've accomplished. First, we've made it much clearer what's in %list and %list2: references to Stringlists. This prevents accidents such as someone accidentally incrementing the value of a %list — the following would not be allowed:
 %list is object stringlist
  ...
 %list = %list + 1 <==Invalid
Nor, would the following:
 %list  is object stringlist
 %list2 is float
  ...
 %list = %list2 <==Invalid
While this might not seem that useful, it can be much more useful in complex applications with lots of different kinds of objects — except for strings and numbers, only object variables of the same class can be assigned to each other. “Wait a minute, what's a class?” you say. A class is simply a name for objects with identical characteristics. In this case, all Stringlists have a similar structure and have a fixed set of things you can do with them (like $lists). The distinction between a class and individual objects is the same as in normal usage — my blue Subaru is an object in the class of cars.
In any case, beyond the protection of accidental assignment mismatches, what else has using an O-O Stringlist instead of a traditional $list bought us? Well, compare:
$listAdd(%list, 'Now is the winter')
with
 %list:add('Now is the winter')
In the former, when the UL compiler hits the $listAdd it has no idea that it's working with a $list. So, to distinguish $listAdd from other functions that might be doing an add of something to something, the word list has to be used as a qualifier in the $function name — it would not do to have the $function called simply $add. But, with the O-O syntax, the compiler first hits the %list and so now “knows” that whatever follows must be something specific to a Stringlist. So, it is sufficient to use the function name Add because it can only mean the Add that applies to Stringlists, not to other classes. This means that
- The O-O statements tend to be shorter, because the extra qualifiers on the function names are unnecessary. This is often true even when the O-O statements use longer, more meaningful function names than the non-O-O equivalents.
- The absence of the extra qualifiers eliminates a lot of the “noise” in O-O statements. Since $lists names usually have the word “list” in them to keep them straight from other variables and since all the $list functions start with “$list”, code the manipulates $lists often turns into the actual functional code swimming in a sea of the word “list”.
Consider, for example, code that subsets, sorts, and then prints the contents of a $list:
$list_print($listSort($listSub(%list, 'foo'), '1,10,A')))
Now, consider the O-O equivalent:
 %list:subset('foo'):sortNew('1,10,A'):print
Not only is the O-O version considerably shorter, it is much easier to follow — again, it can be read left to right. In the $function version, our eyes glaze over at the sea of $lists and nested parentheses. While, this is an extreme example, it can safely be said that code using O-O syntax is almost always tidier and easier to read than the traditional non-O-O syntax.
Unfortunately, unlike using the O-O syntax for intrinsic (String and Float) data, there is no really good way to simply change lines of code, here and there to gradually migrate from using the pseudo-object-orientation of $lists to the full-on “true” object-orientation of Stringlists. Fortunately, many uses of Stringlists are fairly localized so, in many cases, changing an application from using $lists to using Stringlists can be a matter of changing a few dozen lines of code or fewer. While this doesn't provide any game-changing improvements in functionality (there are few things one can do with Stringlists that one can't do with $lists), it will:
- Make code tidier and easier to understand, especially should your organization hire younger programmers who are familiar with object-oriented programming.
- Familiarize you with object-oriented syntax and object behavior.
On this latter point, it is worth mentioning that because Stringlists are true objects, they correct the flakiness that can occur with $lists when the same $listNew (or other $list-creating statement) is executed multiple times. Because the same $listNew statement always returns the same $list identifier (number), situations where a $listNew is excuted multiple times can be buggy or confusing. Executing the same New operation for Stringlists multiple times, on the other hand, returns distinct Stringlist references and so eliminates the bugginess or confusion inherent in the $list equivalent.
It is worth emphasizing that a Stringlist object variable does not contain the Stringlist object any more than a variable used to hold a $list identifier contains the $list itself. In both cases, the variable contains a reference to the underlying object so that assigning one Stringlist object variable to another does not create a copy of the object but simply copies the reference to the underlying object, just as assigning a $list identifier from one variable to another does not copy the $list itself. While, there is a Copy function for Stringlists:
%list is object stringlist %list2 is object stringlist ... %list2 = %list:copy
the assignment of the references is much more efficient, of course, and in many cases, just what one wants, anyway.
Note that Stringlists are no more or less efficient than $lists; in fact, they share most of the same underlying code.
Also, all Stringlist methods are longstring capable. That is, they a accept and return longstring values (strings longer than 255 bytes). For historic reasons, the $list API contains a mix of longstring capable and non-longstring capable functions. Many of the non-longstring capable $functions have extra parameters to deal with the fact that, while $list items could be longer than 255 bytes, they could only be processed 255 bytes at a time. The Stringlist API, written well after longstrings became available, didn't need to deal with any of these issues. This means that the $list API is considerably more complicated than the Stringlist API, without providing any additional functionality. This is just another reason to use the Stringlists in lieu of $lists.
So, in keeping with the theme of this page the recommendations are:
- Find some existing $list code that's relatively localized and convert it to use Stringlist objects.
- Write any new code using Stringlists rather than $lists.
Collections
So far, we have seen how Strings and numbers are simply objects, and can be manipulated using O-O syntax without much effort. We've also seen that $lists were really objects, and that $list identifiers behaved very much like object variables, albeit kludgily because there was no formal object support in User Language. As such, with a modicum of effort, an application that uses $lists could be modified to use Stringlists, tidying up the code without any loss of functionality. As our last exercise in learning object-oriented programming we now look at Collections.
Collections are essentially object-oriented replacements for traditional Model 204 arrays, but, as you'll see, with considerably more functionality.
Model 204 arrays, unlike $lists, are most certainly not objects in the O-O sense because Array variables are not references to array objects, they are the arrays themselves. There is no way, to have two variables referencing the same array (except in the special case of arrays being passed as parameters) and arrays are created at compile-time, unlike objects (and $lists) which are created at run-time. In addition to making arrays non-O-O, these restrictions can be problematic for applications:
- Application coders must determine ahead of time what the largest number of array items will be and allocate space for them, at compile time. This is both wasteful of server space as often the maximum likely array size is likely to be much bigger than the commonly required one, and insufficient, as invariably any arbitrary limit can and will ultimately be exceeded.
- The inability to have multiple references to the same array can make certain code clumsy and unintuitive.
Collections provide a nice object-oriented alternative to arrays. Like arrays, collection variables are declared as being composed of variables of a specific type. For example, an array of numbers might be declared as:
%numbers is float array(20)
A comparable collection declaration would be something like:
%numbers is collection arraylist of float
The word collection, here, means exactly the same thing as object but is used to distinguish collection objects which have certain special characteristics (such as having an Of clause in the declaration) from other objects. Because collections are so common in O-O applications, the keyword collection is actually optional and the above declaration could be more simply written as
%numbers is arraylist of float
Probably, the first thing that jumps out at you is the fact that we don't have to declare (and, in fact, can't) the maximum number of items to be allowed in the Arraylist. This means that Arraylists will only use as much space as is needed for the data at hand and will not add a lot to server size requirements for the odd cases where an arraylist happens to have a lot of items. In fact, collection items are stored in CCATEMP on an as-needed basis, with only the currently referenced item being stored in the server. So, collections have very small server footprints.
One appealing aspect of using an Arraylist is that, by and large, references to arraylist items are identical to references to array items. That is, to set %x to the third item in either an Array or Arraylist one can simply do:
%x = %numbers(3)
And to set the the %i'th item in either an Array or an Arraylist, one can do:
%numbers(%i) = %x
So, all one has to do to switch from using an Array to an Arraylist is to simply change the variable declaration? Well, not quite. Suppose we had something like:
 %cost    is float array(20)
 %ncost   is float
  ...
 o: for each occurrence of cost
    %ncost = %ncost + 1
    %cost(%ncost) = value in o
 end for
Now, we change the above to:
 %cost    is arraylist of float
 %ncost   is float
  ...
 o: for each occurrence of cost
    %ncost = %ncost + 1
    %cost(%ncost) = value in o
 end for
The problem we hit immediately is that the Array version of code never explicitly created the array — the array is created at compile-time. Arraylists, on the other hand, being objects need to be created a run-time — the fancy O-O term for this being instantiation. This is identical to the requirement that a $list be created with a $listNew. Not having created an instance of the Arraylist, we will get a null pointer error in our first reference to the object. We can fix this problem one of two ways. One way is to simply create a new Arraylist instance before using it:
 %cost    is arraylist of float
 %ncost   is float
  ...
 %cost = new 
 o: for each occurrence of cost
    %ncost = %ncost + 1
    %cost(%ncost) = value in o
 end for
But, it is also possible to tell the O-O infrastructure to auotmatically create a new Arraylist the first time it is referenced:
 %cost    is arraylist of float auto new
 %ncost   is float
  ...
 o: for each occurrence of cost
    %ncost = %ncost + 1
    %cost(%ncost) = value in o
 end for
While in many instances this is a bad idea (since it can become unclear in the code where an object will be first created or where it's first intended to be created), experience has shown that, at least for collections, the use of Auto New usually works out fairly well. This is especially the case for applications where an Array was changed to an Arraylist.
But, we hit a second problem with the above code. Specifically, an attempt to set item number%ncost in the Arraylist in the above example will almost cause request cancellation, because that item won't be there, yet. This is really no different from the Array code where, if you tried to add a 21'st item to the Array you would get a request cancellation because the array was declared to have 20 items. However, because the Array is created at compile-time with 20 entries, one can simply set items 1 through 20. When switching the code to use an Arraylist the code will have to be changed, somewhat:
 %cost    is arraylist of float auto new
 %ncost   is float
  ...
 o: for each occurrence of cost
    %ncost = %ncost + 1
    %cost:add(value in o)
 end for
While this makes it a bit more trouble to trivially change from using an Array to an Arraylist, often the populating of the array is isolated to one place so the change is fairly trivial. Even better, it can eliminate the need for a count variable like %ncount since the code could be simply written as:
 %cost    is arraylist of float auto new
  ...
 o: for each occurrence of cost
    %cost:add(value in o)
 end for
“Hold on” you might say. “I need %ncost later on”:
 for %i from 1 to %ncost
    %x = %cost(%i)
    ...
 end for
But, this is unnecessary. One nice thing about collections is you always have a Count function that returns the number of items in the collection so the above code could simply be written as:
 for %i from 1 to %cost:count
    %x = %cost(%i)
    ...
 end for
In addition to eliminating the need for the %ncost variable, this eliminates the chance of a bug where %ncost doesn't accurately reflect the count of items on the Arraylist. Of course, if you have a lot of references to %ncost, you might prefer not to fix them all up, in spite of the benefits. In this case, you could simply leave the incrementing of %ncost in the loop that populates the Arraylist or, if you're compulsive about performance, set it after the loop:
 o: for each occurrence of cost
    %cost:add(value in o)
 end for
 %ncost = %cost:count
So, hopefully, it's clear that with minimal effort, an application can be changed from using an Array to an Arraylist. This would have the immediate benefit of reducing the server size required by the application while, at the same time, reducing any limit on the number of items one might put into the Arraylist. Even better, a wide variety of Arraylist functions become available once you've made this change. There are functions to make copies of Arraylists, to sort them, to search them, to insert or remove items, to find minima and maxima, and much more. In addition, you have the power of multiple references to the same Arraylist. You can even have Arraylists of Arraylists, if that's useful.
All that said, there might be cases where it's non-trivial to convert from using an Array to an Arraylist. Specifically, your application might depend on the fact that the Array is pre-populated with a specific number of items. For example, suppose in the above example, there was an array of 20 items that corresponded to 20 possible product numbers and total costs are calculated as:
o: for each occurence of product.no %cost(product.no) = %cost(product.no) + cost(occurrence in o) end for
Such an application does not lend itself to using the Arraylist Add method as the product numbers won't necessarily arrive in numeric order. One approach would be to pre-populate an Arraylist with zeros:
for %i from 1 to 20 %cost:add(0) end for
But this somehow seems a bit adhoc and dissatisfying. Another approach, for such an application is to use a different kind of collection, a FloatNamedArraylist. A FloatNamedArraylist uses a number to index the items but:
- The item numbers don't have to be added sequentially.
- Negative and fractional indexes are allowed.
Generally, the most useful aspect of FloatNamedArraylists is the first of these items (as it is in our example), but the second can be useful on occasion, too. Using a FloatNamedArraylist, we can now fix up the above example:
%cost is floatNamedArraylist of float auto new ... o: for each occurence of product.no %cost(product.no) = %cost(product.no) + cost(occurrence in o) end for
But, this still isn't quite right. While one can set a FloatNamedArraylist item whether or not it currently has an explicitly set value, by default you cannot retrieve a value of an item that was never set. Since the assignment statement above first retrieves this current value of the item being updated, this is guaranteed to get a request canceling error the first time it's executed. Fortunately, with the addition of a simple method call, the problem can be fixed:
%cost is floatNamedArraylist of float auto new ... %cost:useDefault = true o: for each occurence of product.no %cost(product.no) = %cost(product.no) + cost(occurrence in o) end for
This setting of UseDefault to True means that a reference to an unset item number will return a default value which, for a FloatNamedArraylist is, of course, zero (though it could be set to something else, if needed). In fact, a FloatNamedArraylist, provides a lower impact way of converting an Array to a collection than an Arraylist. A FloatNamedArraylist with UseDefault set to True acts exactly like an Array with essentially no limit on the number of items and no cost for unused item numbers. So, if you had a FloatNamedArraylist with used item numbers 1, 500, and 9999, the FloatNamedArraylist would use no more space than if the used item numbers were 1, 2, and 3. This is what's know as a sparse array. Still, even though a FloatNamedArraylist provides a simpler migration from an Array, it is recommended that in the common case where the array/collection items are sequentially added, that an Arraylist be used, as this provides a more natural way of representing the data, even though it might be a little more work.
Hopefully, you'll have noticed something unusual in the above code, specifically that while UseDefault looks like a function or subroutine invocation, it's actually being set. This is something called a property though it can also be thought of as a variable, too, much as images have variables that can be set. In any case, objects can have properties or variables with values that can be set or retrieved. The syntactic magic that allows the above True might also catch your attention. A UL programmer would guess that it's a field reference, but it's not — it's simply a boolean value. Hopefully, it seems natural enough that, for now, you're willing to accept and understand its meaning it without fully understanding the syntax.
As a last note about FloatNamedArraylists, it is possible to loop through only the items that have actually be set in the request. For example, the following, prints the index and value for all set FloatNamedArraylist items in %cost:
 %cost     is floatNamedArraylist of float
 ...
 for %i from 1 to %cost:countries
    print %cost:nameByNumber(%i) with ': ' with %cost:itemByNumber(%i)
 end for
The “name” of a FloatNamedArraylist item is the index, so that if you had updated items 1, 73, and 9999, NameByNumber would return those numbers for input parameters 1, 2, and 3, respectively and, of course, ItemByNumber would return the value to which these items had been set.
Hopefully, by this point, you see:
- How easy it is to change from using Arrays to collections.
- The power that such a change provides.
Collections are a big topic and there are many very useful sub-topics that haven't been covered here:
- The ability to have collections of any datatype such as Strings, Longstrings, or even objects such as Stringlists or other collections.
- NamedArraylists which provide collections where the index is a string. They provide User Language with associative array capability.
The magic Item method
In the converting an Array to a collection, it's critical that a collection item can be referenced in a syntactically identical way to an Array, namely with %array(<subscript>). Not only is this incredibly useful for such a conversion but it seems very logical and natural; so much so that one might not give it a second thought.
However, in the object-oriented paradigm, the contents of an an object are always accessed by a method or variable name — in User Language via <object>:<member> syntax. So, what gives? Well, what gives is that for collection objects, because the most common operation for a collection and the whole reason for the collection's existance is the ability to extract individual items from the collection, there is a default method if one isn't specified explicitly, namely the Item method. This means that if %foo is an Arraylist, %foo(4) is functionally identical to %foo:item(4). Which one is used is largely a matter of taste though, again, the fact that %foo(4) seems so natural suggests that there is generally no particularly good reason to explicitly specify Item.
In addition to collections, certain “collection-like” objects such as Stringlists can also use the implied Item method. Stringlists are not, strictly speaking, collections, since they cannot be made up of arbitrary datatypes (they have no Of clause on their declarations). However, for most purposes, the behave very much like an Arraylist of strings and so are collection-like. As such, if %sl is a Stringlist object, %sl(%i) is functionally identical to %sl:item(%i).
Conclusion
Hopefully, after reading this page and following its suggestions, you're now much more comfortable with object-oriented programming syntax and have a sense of how objects and object variables are used. This is half the battle in becoming an object-oriented programmer. The other half is writing your own classes (types of objects). But, it is possible to use object-oriented technology without ever writing your own classes. In fact, it's a good bet that many, and maybe even most programmers in other O-O languages write few if any of their own classes.
Sirius Software provides a wide variety of built-in (System) classes to increase the power of the User Language programmer and to give User Language programmers plenty of practice writing O-O code. So, even if you never write your own classes, it is well worth moving to using object-oriented syntax and facilities as much as possible. Ultimately, you will get comfortable enough with the concepts that you will recognize the situations where you want to write your own classes and will have a sense of how those classes should be structured. If/when that happens, you will be ready to take the next step. Until then, you can already consider yourself an object-oriented programmer (though not an expert or “guru”).