Chapter 9

Working With Collections

Much of the work programmers do in the Telescript environment is wrangling objects--grouping objects together, ordering objects within a group, adding objects to a group, finding and removing objects from a group, and performing many other group manipulations. To help you wrangle, Telescript provides a set of predefined classes that define and work with groups of objects. They are the class family Collection along with its minions List, Set, BitString, OctetString, Stack, String, and Dictionary.

This chapter takes a close look at Collection and its subclasses. You'll learn how to derive classes from the Collection class families, and then see how to instantiate those classes to create collections, lists, sets, stacks, and other Collection objects. A look at each class's features shows you how to manipulate these objects--how to modify and examine their contents and how to convert them to objects of other types.

An Overview of Collection Classes

The following class hierarchy shows the inheritance relationships between Collection and other classes described in this chapter. All class families have double square braces following their name ([]). The mix-in superclass Compared is preceded by <: because the superclass can be specified as Compared or either of its two subclasses--depending on how a class is derived from the class family Collection. (You'll read about the details of Collection derivation later in this chapter.) All classes and class families described in this chapter are in boldface.

Object

* Collection[] (<:Compared)
* * List[] (Ordered)
* * * BitString
* * * OctetString
* * * Stack[]
* * * String (Cased)
* * Set[] (Verified)
* * * Dictionary[]

The Collection Classes

As you can see from the inheritance tree, the class family Collection defines features that are inherited by seven other predefined Telescript classes and class families--the collection classes.

A Collection object is an unordered group of objects. The objects are the collection's items; you provide them (or none, if you prefer) when you initialize the object. Once a collection exists, you can use its operations to examine and modify it--check to see how many items there are, add a new item, delete an existing item, check to see if a given item is included in the collection, or take any number of other possible collection actions.

A List object inherits the features of Collection and adds a key aspect: it orders its items so they each have a numbered position--1, 2, 3, 4, and so on. It adds new operations that can add, delete, and move items within the list; operations that can retrieve an item from a specified position; and operations that can add sublists (other lists of items) to the list.

List has four built-in subclasses. BitString creates objects that store lists of 1s and 0s--in other words, that store binary data. OctetString creates objects that store strings of octets; you can think of an octet string as a stream of bytes. Stack creates a LIFO (Last In, First Out) stack where you can push, pop, swap, and roll items. And String creates objects that store lists of characters. String isn't discussed in this chapter; you can read about it in the next chapter, where it plays an integral role in the text-handling capabilities of the Telescript environment.

Set is the second of Collection's two built-in predefined subclasses. Unlike List, it does not impose order on its items. Instead, it ensures that each of its items is unique--that no two items in the set match. A Set object, then, is a collection of unique items that may occur in any order. Set operations allow you to work with one set against another: to create a union of the two sets, to create the intersection of the two sets, to create a set that is the difference of one set from the second set.

Set has one built-in subclass: Dictionary. Dictionary treats each of its items as a key. It pairs each key with a value--that is, another object (you can specify the type of object allowed). A Dictionary object, then, contains key-value pairs. Dictionary's operations allow you to provide a key and retrieve the value associated with that key. The operations also allow you to add and drop key-value pairs and modify the dictionary in other ways. Dictionaries are commonly used in the Telescript environment as an unordered lookup table. The dictionary groups together a set of values (whatever objects are provided in initialization and later modification); each value has a key used for retrieval. The characteristics that Dictionary inherits from Set ensure that each key is unique.

Classes Associated With Collection Classes

The collection classes have other classes associated with them. Some of these classes are mix-ins used to define the collection classes, others are flavors that define objects used to work with collections. The following class tree shows the inheritance of these classes:

Object

* Iterator[]
* Exception
* * ProgrammingException
* * * CollectionException
* * * * KeyInvalid
* * * * PositionInvalid
* * * * StackDepleted
Compared

* Equal
* Same
Ordered

Verified

Iterator defines an object that can produce a reference to an item in a collection each time it's requested to do so. An iterator is returned by the Collection operation iterator; you can use the iterator to step through the collection's items. The iterator allows you to examine the contents of a collection.

CollectionException is a sub-subclass of Exception; it defines three subclasses--KeyInvalid, PositionInvalid, and StackDepleted--that are thrown by operations of Collection and its subclasses.

Compared is a mix-in class that defines how two objects are considered matching objects. Its two subclasses--Equal and Same--use different criteria to determine how objects match. One of these three mix-ins--Compared, Equal, or Same--is chosen as the mix-in superclass for Collection when you derive a class from it. The mix-in chosen determines how matching works in any operations that depend on matching.

Ordered is a mix-in class used as a superclass for List. As you learned in Chapter 5, it's also used extensively with primitives such as numbers, bits, and characters, where it determines the order relationship between two objects--before, equal to, after, or unordered. By mixing Ordered into the List definition, it allows two lists to be ordered so one list can be before, equal to, or after another.

Verified is a mix-in class used as a superclass of Set. It's used to help determine if a set is internally inconsistent--that is, outside influences (discussed later in this chapter) have created matching items in the set.

Class Families and Inheritance

Now that you've been introduced to collection classes and their associated classes, it's time to take a close look at how inheritance works in the Collection family line. It's a good chance to see how class families pass on characteristics to their subclasses, and how formals can be used to define a class family's superclasses.

If you look at the first class tree presented in this chapter, you'll notice that the class family Collection has two subclasses that are also class families: List and Set. List has in turn some subclasses that are flavor classes, and one subclass that is a class family. To see how this works, consider the nature of a class family.

A class family, as you'll recall from previous chapters, is a class-producing function that specifies classes within itself using formal parameters (formals). When these formals are satisfied by actual parameters (actuals), the class family derives a class. A class family itself can't be directly instantiated; you must first derive a class from the family and then instantiate the derived class.

Inheriting From a Derived Class

Because a class family isn't technically a class, it may seem at first thought that a class family can't have subclasses. On further reflection, you might see that you can first derive a class from a class family and then create a subclass of the derived class. That's what happens with the class family List and the class BitString (both shown in the previous hierarchy). BitString is really a subclass of the derived class List[Bit, Equal]. The class hierarchy doesn't show the derived class, but if you look up BitString in the class reference documentation, you'll see in the class signature that List[Bit, Equal] is the primary superclass of BitString. In other words, BitString's class definition defines the derived class List[Bit, Equal] as its primary superclass.

Inheriting From a Class Family

Now consider the class family Stack shown in the class tree. If you look at its definition in the class reference documentation, you'll see that Stack, like BitString, specifies a derived class of List as its primary superclass. But--as a class family--Stack can use a formal anywhere in its definition where it specifies a class. One place you'll always find classes in a class definition is in its superclass specification. In this case, Stack defines its own formals Item and Match. It then uses those formals to specify its superclass, which is a class derived from the class family List. The formals are supplied as the actuals of List in this way: List[Item, Match].

The effect is that Stack's inheritance isn't determined until you derive a class from Stack. When you supply actuals for Stack, those actuals are passed on as actuals for List to derive a primary superclass for Stack.

An Example

If any of this seems confusing, consider another example. The class family Stack requires actuals for its formals Item and Match. If you provide the actuals Number and Equal, you create the derived class Stack[Number, Equal]. This derived class uses your actuals to satisfy its formals wherever they occur in the class family definition. Two of those formal occurrences are in the inheritance specification for the class family. These formals specify which derived List class will be the immediate superclass of Stack. When they're satisfied, they specify the derived class List[Number, Equal] as the immediate superclass of Stack[Number, Equal].

As a further example, if you provide the actuals Primitive and Same for Stack, you create the derived class Stack[Primitive, Same]. It has the immediate superclass List[Primitive, Same] because the actuals are used to define the immediate superclass.

In cases of inheritance like the one just described, it may help to think of the formals of a subclass family (like Stack) as "percolating formals." They are linked directly to the formals of the superclass family, so any actuals you supply percolate up to derive a class from the superclass family.

Percolating formals can go up many levels, as is the case for Stack. When you supply actuals for the formals Item and Match, those actuals are passed up to derive a class from the superclass family List. The derived List class, in turn, passes those actuals up to its superclass family Collection to derive a class there. So when you derive the class Stack[Number, Equal], your actuals Number and Equal are passed on to derive the superclass List[Number, Equal] and the super-superclass Collection[Number, Equal].

If you want all the gnarly and incestuous details of inheritance and class families, you'll find them in Chapter <<<yet to be written>>>, which gives full information on defining your own class families.

The Collection Class Family

The class family Collection is the root of all collection classes. It's a concrete and unsealed class family, so you can derive classes and instantiate them to create Collection objects. You can also define your own Collection subclasses.

As you read earlier, Collection defines an object that is an unordered group of objects, the collection's items. The number of items in the collection determines the collection's length. For example, a collection that includes 25 items has a length of 25; a collection with no items has a length of 0. A collection's length is unbounded; a collection can have as many items as available computing resources will permit.

It's important to remember that a collection's items are unordered. None of a collection's operations depends on order when finding, retrieving, and adding items to the collection. In fact, if you iterate through a collection's items, you can't be assured that the items will be iterated in the same order each time. If you want ordered items, you can work with List, a subclass of Collection.

Because Collection is a class family, you must derive a class from it before you can instantiate a collection. To do so, you must satisfy two formals: Item, which constrains the type of items that may be included in the collection; and Match, which specifies the mix-in superclass that determines how the collection matches items. If you use Collection as a class name without providing actuals, the default derived class is Collection[Object, Compared].

Specifying the Type of Collection Items

To specify the type of item that may be included in a collection, you provide a class for the formal Item. Any items you add to the collection must be a member of the specified class. If, for example, you specify Number as an actual for Item, the collection can include any items that are integers or real numbers. If you use Object as the actual for Item, any type of object may be included in the collection. Object is Collection's default actual, so a default collection can include any type of object.

Specifying the Way a Match Is Determined

Much of a collection's business depends on matching--finding an item that matches an argument, for example, or eliminating matching items from a collection. To determine a match between two objects, a collection uses an operation inherited from its Compared mix-in: compare. compare accepts two objects as its arguments and compares them to determine their relationship: equal or unordered. Any two objects that are equal are considered a match; any unordered objects are not a match.

(Don't confuse the mix-in Compared with the mix-in Ordered that you read about in Chapter 5. Ordered allows you to determine a relationship between two ordered objects such as most primitives: equal to, before, after, or unordered. Compared lets you determine a relationship between two objects whether or not they're ordered objects, and--as it's implemented--determines only two relationships: equal or unordered.)

The Telescript language uses two kinds of matching: copy-equal matching and same-object matching. In copy-equal matching, two objects are considered equal (a match) if they are copy equal--that is, one object could have been created by calling the copy operation on the other object. (copy is an operation inherited by all objects from the root class Object. It's discussed in Chapter 5 and in greater detail in Chapter <<<yet to be written>>>.)

In same-object matching, two objects are considered equal (a match) only if they are references to the same object. If, for example, the variable tempos and the variable speeds both refer to the same collection, tempos and speeds would be considered a same-object match. Same-object matching is much more restrictive in its definition of matching objects than copy-equal matching. Two objects that may be copy equal aren't necessarily a same-object match; two objects that are a same-object match, however, must always be copy equal.

Keep in mind, when considering same-object matching, that Telescript primitives (numbers, characters, etc.) are the same object if they have the same value. In other words, the integer 5 used in one variable assignment is the same object as the integer 5 used in another variable assignment, even if you might think of them as different instances. The Telescript engine, for economy, has a single instance of 5 that it uses for both cases of 5.

Collection's Match formal specifies how matching is determined within a collection. The formal is defined to accept Compared or either of its subclasses Equal or Same. The class supplied as the actual becomes the mix-in superclass of the derived class. So if, for example, you derive a Collection class by specifying Equal as the Match actual, the class's flavor superclass is Object and its mix-in superclass is Equal. (The flavor superclass is always Object no matter what mix-in superclass is specified.)

The three possible Match mix-in classes each have a definition of how the relationship "equal" is determined when the operation compare is called.

The Compared Mix-In

Compared is the superclass of both Equal and Same; it defines and implements the compare operation available in all three mix-in classes. compare, as implemented by Compared, takes two objects as arguments and returns their relationship:

The Equal Mix-In

Equal has the same interface and implementation as Compared. When you use its compare operation to compare two objects, it returns equal for copy equal objects and unordered for object that aren't copy equal. There's no practical difference between specifying Equal or Compared as the Match actual for a collection. Both mix-ins set the collection to consider copy-equal objects as matching.

Equal is the most commonly used class for Match--so most derived collection classes are defined to consider copy-equal objects as matching.

The Same Mix-In

Same has the same interface as Compared, but has its own implementation of the operation compare. This implementation is almost identical to Compared and Equal's implementation with one important difference: it returns the identifier equal for two objects only if those objects are references to the same object. It returns the identifier unordered for any two objects that aren't references to the same object.

The functional difference is that if you derive a collection class using the mix-in Same, matches are made using same-object matching, not the copy-equal matching used for classes derived using Equal or Compared.

Initializing a Collection

To create a Collection object, you must first derive a class from Collection by supplying actuals for Item and Match and then provide initialization arguments. This all takes place in an initialization statement.

Collection's class signature (shown in the class reference documentation) shows you the formals and how they must be satisfied:

Collection: interface[Item: Class; Match: Class<:Compared]
In other words, Item can be satisfied with any class, while Match must be satisfied with Compared or either of its two subclasses, Equal and Same--a topic you just read about.

The initialize operation's signature shows you the required initialization arguments:

initialize: op (items: Item ...);
The argument Items is followed by an ellipsis (...), so this means you can supply as many arguments as you wish (or none if you don't want to supply any arguments). The constraint for items is given as Item, which--as you'll remember--is one of the formals for Collection. Whatever actual you supply for Item is the constraint for the argument items. In other words, if you derive the class Collection[Number, Equal], the formal Item is satisfied by the actual Number. This means that the constraint for items is Number.

Each of the items arguments you supply for initialization becomes an item of the collection. Consider this example:

Collection[Number, Equal](5, 3.78, 9375, 0.05760);
This initialization statement derives the class Collection[Number, Equal], which can only include items that are members of the Number class and that considers objects matching if they're copy equal. The four arguments that follow are all items of the collection. Once the collection is initialized, you have a collection with the items you specified as initialization arguments.

Modifying a Collection

Collection offers three operations that let you modify a collection:

Examining a Collection

Collection also offers features that you can use to examine the length and contents of a collection:

Using an Iterator

An iterator is an object defined by the class family Iterator. An iterator holds a reference to a corresponding collection, and can produce references to each of the collection's items. Like a collection, an iterator is created as an instance of a class derived from a class family. The class family Iterator has a single formal to satisfy: Item. This formal has the same meaning here as it does in the class family Collection; it constrains the type of items to which the iterator may refer. The derived class Iterator[Primitive], for example, defines an iterator that may only refer to items that are primitives.

Iterator is an abstract class family; you can't create instances of it yourself, but the Telescript engine can--and does--whenever you call iterator on a collection. The iterator returned is constrained to the same type of items as the collection from which it was returned. The iterator has a reference to the collection for which it was created, and can produce a reference to each of the collection's items. In this way the iterator can refer to the same group of items as the collection.

An iterator offers three attributes: current, next, and isDone. You use these attributes to examine a collection's items. The first time you use the next attribute, it returns one of the collection's items. The next time you use next, it returns another of the collection's items. In subsequent uses of next, it returns the rest of the collection's items without repeating any item. When it has returned all of the items in the collection it returns nil.

If you want to repeat access to a collection item, use the current attribute. It returns the last item produced by the next attribute. You can use it again and again without affecting the sequence of iteration used by next.

You can check the isDone attribute at any time to see if the iterator has returned all of a collection's items. Its value is true if that's the case, false if not. This is particularly useful if the next operation has returned nil--the nil could either be an item in the collection, or it could mean that the iterator has returned all the items in the collection. isDone tells which case is true.

When an iterator is done, its current and next attributes will return only nil. If you need to iterate through a collection's items once again, you should discard the iterator, call iterator on the collection again, and work with a new iterator.

It's important to note that an iterator is used strictly for examination. Its items are references to existing items in a collection, and using an iterator has no effect on the collection from which it sprang. Discarding an iterator has no effect on the collection.

It's also important to note that--because a collection is an unordered group of items--you can't depend on iterators to iterate through a collection's items in the same order each time you use them. (If you use an iterator for a list, though, it does step through items in ascending order.)

Copying and Converting a Collection

You can copy a collection if you want using either of two operations:

If you want to convert a collection to a list, you can use the conversion operation asList. It returns a list whose items are those of the collection. Because a collection is an unordered group of items, you can't be sure what the order of the items in the returned list will be.

Example Code

The following short Telescript program shows how you can instantiate, modify, and use an iterator to examine a collection. (The lines are numbered here for easy discussion later.)

1	do{
2		/* Initialize a collection and modify it. */

3		grabbag: = Collection(1, 'A', 4.657, 'A', 'z');
4		"The collection after initialization:".dump();
5		grabbag.dump();

6		grabbag.exclude('A');
7		"After an A has been excluded:".dump();
8		grabbag.dump();

9		grabbag.include(6);
10		"After a 6 has been included:".dump();
11		grabbag.dump();

12		/* Create an iterator for the collection "grabbag", then iterate
13		   through its items in a while loop that tests the iterator's
14	 	  isDone attribute. */

15		look: = grabbag.iterator();

16		"An iterator going through the items of the collection:".dump();
17		while !look.isDone {
18			look.next.dump()
19		};
20		"There are no more items to be iterated.".dump();
21	}
When you execute the program, you get the following results:

String: <The collection after initialization:>
Collection: <5 elements>
        Integer: <1>
        Real: <4.657>
        Character: <65(A)>
        Character: <65(A)>
        Character: <122(z)>
String: <After an A has been excluded:>
Collection: <4 elements>
        Integer: <1>
        Real: <4.657>
        Character: <65(A)>
        Character: <122(z)>
String: <After a 6 has been included:>
Collection: <5 elements>
        Integer: <1>
        Real: <4.657>
        Integer: <6>
        Character: <65(A)>
        Character: <122(z)>
String: <An iterator going through the items of the collection:>
Integer: <1>
Real: <4.657>
Integer: <6>
Character: <65(A)>
Character: <122(z)>
String: <There are no more items to be iterated.>
The program is pretty straightforward. It creates a collection named grabbag and initializes it with five items, then dumps the collection so you can see its contents. Notice that two of the items are duplicates: the "A" characters. Notice also that no actuals were specified in the initialization, which derives the class Collection[Object, Compared] (the default derivation) from the Collection class family.

Line 6 excludes an "A" from the collection; line 9 includes a 6. A dump of the collection after each action shows the new items.

In line 15, the operation iterator is called on the collection to create an iterator named look. A while loop checks the iterator's isDone attribute; as long as it's false, the loop steps through the items of the iterator (which are the collection's items). When there are no more items to step through, isDone becomes true, and the loop ends.

The List Class Family

The List class family adds an important element to Collection: order. Each of the items in a list is associated with an integer that is the item's position. Positions start at a value of 1 and extend to a value that equals the length of the list. Position is important when working with a list; you use it to specify items and ranges of items within a list. You may also receive the position of an item as the result of an operation. (If you program frequently in C, be sure to remember that Telescript lists start with position 1, not with position 0 as C arrays do.)

The List class family, like Collection, has two formals: Item and Match. These formals have the same meaning they do in the Collection class family, and are, in fact, "percolating formals." The actuals you provide to derive a List class are used as actuals to derive a Collection superclass.

List's definition also specifies a mix-in superclass--Ordered--that allows two lists to be compared and their relationship evaluated. One list may be before, after, equal to, or unordered in relation to the other list.

Initializing a List

To create a List object, you must first derive a class from List by supplying actuals for Item and Match. You then provide initialization arguments. You do both these steps in an initialization statement that looks like the initialization statement for a collection. The actuals you supply have the same constraints as the actuals for Collection. And you supply variable arguments for initialization just as you do when initializing a collection. All the arguments you supply become items of the newly initialized list. The only difference is that the order of the initialization arguments is important: the order of the arguments becomes the order of the items in the list. The order of a list's items determines each item's position in the list.

Modifying a List

A list offers two sets of operations for modifying the list: operations for working with single items, and operations for working with lists and sublists. Most of these operations require you to specify the positions of one or more items within a list.

Specifying a Position

To specify the position of an item in a list, simply provide the integer of the position that corresponds to the item--3, for example, for the third item of a list. It's illegal to specify a position that's less than 1 or greater than the length of the list (unless you're appending items to a list using the add operation).

Working With Single Items

A list inherits two operations from Collection that it can use to work with single items: include and exclude. Both of these operations have been specialized for the class family List; they take into account the fact that the items of a list are ordered. If you use include to add an item to a list, the item appears at the end of the list. And if you use exclude to remove an item from a list, the positions of all items following the removed item are decreased by 1 so there's no gap in the position numbering.

List defines five of its own operations for working with single items:

The Telescript language provides a special syntax for using the set operation on any class that defines set with the required signature. This syntax works on an instance of List or any of its subclasses. The syntax is an assignment expression that takes this form:

<object>[<position>] = <item>
Using this syntax makes setting an item of a list feel like an assignment. This expression, for example,

scores[5] = 235
sets the fifth item of the list scores to a value of 235. The expression is equivalent to this feature expression:

scores.set(5, 235)

Working With Lists and Sublists

List defines two operations that work with lists and sublists. (A sublist is a range of zero or more items within a list.) To specify a sublist within a list, you provide two positions: the starting position and the ending position. The starting position is the position of the first item in the sublist; the ending position is the position of the item following the last item in the sublist (which is length + 1 if the last item of the sublist is the last item of the list). To put it mathematically, the sublist includes items whose positions are in the interval [start, end).

Most sublist specifications have an ending position that is larger than the the starting position. Anytime this is the case, the sublist has one or more items. For example, a sublist with starting and ending positions of 3 and 7 specifies items 3-6, a total of four items. And a sublist with the starting and ending positions of 7 and 8 specifies item 7, a total of one item.

You can also specify a sublist whose starting and ending positions are equal. This sublist contains no items and is located just before the start position. In other words, equal starting and ending positions specify a single position that has no items. This is different than the position you specify for a single item because it falls between items and not on an item. For example, if you specify the position 5, you specify item 5 in the list. If you specify the sublist 5,5, you specify the position between item 4 and 5--but you don't specify any item in the list.

A sublist with no items is an important tool when you use the splice operation (described later) because it allows you to specify a position between two items when you insert one list into a second list. If you want to append the list, you can specify the substring length+1, length+1 which specifies the position just after the last item in the list. And if you want to prepend the list, you can specify the substring 1,1 which specifies the starting position just before the first item in the list.

When you specify a sublist, note that it's illegal to use starting and ending positions that are less than 1 or greater than length + 1. It's also illegal to specify a sublist whose ending position is less than its starting position. You can, if you want, use nil as the starting or ending position of a sublist. Nil in the starting position specifies the position 1. Nil in the ending position specifies the position length + 1. This means that the substring specified nil, nil defines all the items from the first item of the list to the last item of the list, inclusive--in other words, the whole list.

List's two operations that work with lists and sublists are these:

You can think of splice as an operation that can snip out a part of a list and return it to you, splice in a new list of items to a list, or do both simultaneously. You can supply a nil as the list argument if you wish, which allows you to remove a sublist from a list without inserting new items. Or you can use the same position for both position arguments to insert a new list into the responding list without removing any existing items.

Examining a List

List inherits examination features from Collection: the length attribute and the operations examine, iterator, and shallowCopy. iterator has been specialized so the iterator it returns iterates through the list's items in ascending order from position 1.

List also provides two examination operations of its own:

You can use get, like set, through a special syntax that may be used on any instance of List or its subclasses. The syntax is an expression that takes this form:

<object>[<position>]
It evaluates to the value of the item at the specified position. This expression, for example,

scores[5]
gets the fifth item of the list scores. The expression evaluates to the value of that fifth item. It's equivalent to this feature expression:

scores.get(5)

Comparing Lists

List's mix-in superclass Ordered adds the operations maximum, minimum, and order to a list. These operations determine and act on the order relationship between two objects. List specializes the order operation so that all three oiperations can make sense of lists, which usually have multiple items. Here's how the operations decide if one list is before, after, equal to, or unordered in relationship to a second list:

The items of one list are compared one by one, in order, with the items of the second list. Comparison starts with position 1. If the two items there are equal, the comparison moves to the next position, and continues on to successive positions until an unequal pair of items is found. The order of that unequal pair of compared items determines the order of the two lists. If, for example, you compare a first list of (4, 2, 7, 5, 9) to a second list of (4, 2, 6), the lists are equal until position three, where the 7 of the first list is after the 6 of the second list. This means that the first list is after (greater than) the second list.

If a shorter list has items that are equal to the corresponding items of a longer list, then the shorter list is before the longer list. For example, the list (4, 2, 7) comes before the list (4, 2, 7, 5, 9).

If two lists have equal lengths and contain items that are equal to each other and are in the same order, the two lists are equal. For example, the list (4, 2, 7, 5) is equal to the list (4, 2, 7, 5).

Example Code

The short Telescript programs that follow show how you can initialize and modify lists.

do{
	/* Initialize a list and modify it in different ways. */

	letters: = List[Character, Equal]
		('A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J');
	"The list after initialization:".dump();
	letters.dump();

	letters.add(5, 'X');
	"After an X has been added at position 5:".dump();
	letters.dump();

	letters.drop(8);
	"After the item at position 8 has been dropped:".dump();
	letters.dump();

	letters[5] = 'Z';
	"After item 5 has been set to Z:".dump();
	letters.dump();

	letters.reposition(5, 7);
	"After item 5 has been repositioned to position 7:".dump();
	letters.dump();

	letters.transpose(1, letters.length);
	"After the first and last items have been transposed:".dump();
	letters.dump();
}
When you execute the program, you get the following results:

String: <The list after initialization:>
List: <10 elements>
        Character: <65(A)>
        Character: <66(B)>
        Character: <67(C)>
        Character: <68(D)>
        Character: <69(E)>
        Character: <70(F)>
        Character: <71(G)>
        Character: <72(H)>
        Character: <73(I)>
        Character: <74(J)>
String: <After an X has been added at position 5:>
List: <11 elements>
        Character: <65(A)>
        Character: <66(B)>
        Character: <67(C)>
        Character: <68(D)>
        Character: <88(X)>
        Character: <69(E)>
        Character: <70(F)>
        Character: <71(G)>
        Character: <72(H)>
        Character: <73(I)>
        Character: <74(J)>
String: <After the item at position 8 has been dropped:>
List: <10 elements>
        Character: <65(A)>
        Character: <66(B)>
        Character: <67(C)>
        Character: <68(D)>
        Character: <88(X)>
        Character: <69(E)>
        Character: <70(F)>
        Character: <72(H)>
        Character: <73(I)>
        Character: <74(J)>
String: <After item 5 has been set to Z:>
List: <10 elements>
        Character: <65(A)>
        Character: <66(B)>
        Character: <67(C)>
        Character: <68(D)>
        Character: <90(Z)>
        Character: <69(E)>
        Character: <70(F)>
        Character: <72(H)>
        Character: <73(I)>
        Character: <74(J)>
String: <After item 5 has been repositioned to position 7:>
List: <10 elements>
        Character: <65(A)>
        Character: <66(B)>
        Character: <67(C)>
        Character: <68(D)>
        Character: <69(E)>
        Character: <70(F)>
        Character: <90(Z)>
        Character: <72(H)>
        Character: <73(I)>
        Character: <74(J)>
String: <After the first and last items have been transposed:>
List: <10 elements>
        Character: <74(J)>
        Character: <66(B)>
        Character: <67(C)>
        Character: <68(D)>
        Character: <69(E)>
        Character: <70(F)>
        Character: <90(Z)>
        Character: <72(H)>
        Character: <73(I)>
        Character: <65(A)>
As you can see, the program created a list of characters (in alphabetical order so you can more easily see list changes), then modified the list by working with individual items.

The following short program works with substrings:

1	do{
2		/* Initialize a list and modify it using substrings. */

3		letters: = List[Character, Equal]
4			('A', 'B', 'C', 'D', 'E', 'F', 'G');
5		"The list after initialization:".dump();
6		letters.dump();

7		letters.append(List[Character, Equal]('H', 'I', 'J'));
8		"After the string HIJ has been appended:".dump();
9		letters.dump();

10		removed: = letters.splice(4, 7, List[Character, Equal]
11			('X', 'Y', 'Z'));
12		"After the substring XYZ has been spliced into 
13			position 4-6:".dump();
14		letters.dump();
15		"This substring was removed by the splice:".dump();
16		removed.dump();
17	}
When you execute the program, you get the following results:

String: <The list after initialization:>
List: <7 elements>
        Character: <65(A)>
        Character: <66(B)>
        Character: <67(C)>
        Character: <68(D)>
        Character: <69(E)>
        Character: <70(F)>
        Character: <71(G)>
String: <After the string HIJ has been appended:>
List: <10 elements>
        Character: <65(A)>
        Character: <66(B)>
        Character: <67(C)>
        Character: <68(D)>
        Character: <69(E)>
        Character: <70(F)>
        Character: <71(G)>
        Character: <72(H)>
        Character: <73(I)>
        Character: <74(J)>
String: <After the substring XYZ has been spliced into
                position 4-6:>
List: <10 elements>
        Character: <65(A)>
        Character: <66(B)>
        Character: <67(C)>
        Character: <88(X)>
        Character: <89(Y)>
        Character: <90(Z)>
        Character: <71(G)>
        Character: <72(H)>
        Character: <73(I)>
        Character: <74(J)>
String: <This substring was removed by the splice:>
List: <3 elements>
        Character: <68(D)>
        Character: <69(E)>
        Character: <70(F)>
A few things to notice in this last program: the list arguments supplied in lines 7 and line 10 are of the same derived class (List[Character, Equal]) as the class being modified. That's because the compiler won't accept substrings of a different derived class than that of the modified string. In line 10, the positions 4 and 7 are used to specify the substring in positions 4 through 6; remember that the end of a substring is always specified with the position following the last item of the substring.

The List Subclasses

List has four predefined subclasses: BitString, OctetString, Stack, and String. String is discussed in the next chapter. Let's look at the other three subclasses now.

The BitString Class

BitString is a subclass of the derived class List[Bit, Equal]--which means that it's a list designed to include bits as items. It inherits all the features of List, and adds a few new twists of its own.

Bit-String Literals

A bit-string literal, like a primitive literal, is a literal expression that creates an object based on the value in the expression--without using a constructor expression. In this case, it creates a bit string. A bit-string literal starts with a percent sign and a double quote (%"), follows with zeroes and ones, and ends with another double quote ("). For example, the bit-string literal %"10011" creates a bit string whose items are %1, %0, %0, %1, and %1.

It's important to note that when you use a literal to create a bit string, the bit string returned is a protected bit string--that is, it can't be modified. Protection is an advanced topic covered in later chapters. For now, it's enough to know that you can't modify the contents of a bit string literal or use the bit string as an argument for an operation that requires an unprotected bit string (whose constraint is preceded by the keyword unprotected). You must initialize a bit string instead, which returns an unprotected bit string. Or you can copy the string literal and use the copy as the argument.

Initializing a Bit String

When you initialize a bit string, you supply any number of bits and bit strings as arguments. The bits and bit strings--in the order in which they're supplied--provide the items of the new bit string. You can supply bits one at a time (%1, %0, %0, %1, %1, for example), you can supply one or more bit-string literals or variables, or you can combine bits and bit strings as arguments. Combining allows you to concatenate existing bits and bit strings into a new bit string. For example, this constructor expression

BitString(%1, %"10010", %0, %1, %"10101010")
constructs the new bit string %"1100100110101010".

Ordering Bit Strings

BitString specializes its inherited operation order so that when two bit strings are compared, the shorter of the two bit strings is treated as though it had enough zero bits prepended to it to equal the length of the longer bit string. This means, for example, that the bit string 1011 is before 101010 because 001011 is compared to 101110 to determine their order. (If the two bit strings were compared simply as lists of bits, 1011 would be after 101010 because the fourth bit of 1011 is greater than the fourth bit of 101010.)

Conversion

BitString provides its own conversion operation: asOctetString. The operation returns an octet string that is the hexadecimal equivalent of the binary value contained by the bit string.

The OctetString Class

OctetString is a subclass of the derived class List[Octet, Equal]. It's a list designed to include octets (hexadecimal values) as its items. It inherits all the features of List and adds features of its own. We'll discuss all of them except the operation decode, which is an advanced feature discussed in Chapter <<<yet to be written>>>, where you learn about encoding and decoding.

Octet-String Literals

An octet-string literal is a literal expression that creates an octet-string object based on the value in the expression. An octet string literal starts with a dollar sign and a double quote ($"), follows with pairs of hexadecimal digits (0-f or 0-F), and ends with another double quote ("). For example, the octet-string literal $"f4401dba" creates an octet string whose items are $f4, $40, $1d, and $ba.

An octet-string literal, like a bit-string literal, returns a protected octet string that you can't modify. If you need an unprotected octet string that you can modify or use as an unprotected argument for an operation, initialize an octet string to create an unprotected octet string.

Initializing an Octet String

You initialize an octet string by supplying any number of octets and octet strings as arguments. The octets and octet strings--in the order in which they're supplied--provide the items of the new octet string. You can supply the octets one at a time ($F4, $40, $1D, $BA, for example), you can supply one or more octet-string literals or variables, or you can combine octets and octets strings as arguments. Combining allows you to concatenate existing octets and octet strings into a new octet string as in the following constructor expression:

OctetString($50, $"df98", $ab, $"f284")
The expression constructs the new octet string $"50df98abf284".

Ordering Octet Strings

OctetString specializes its inherited operation order so that when two octet strings are compared, the shorter of the two octet strings is treated as though it had enough zeroes prepended to it to equal the length of the longer octet string. This means, for example, that the octet string 9d2 is before 2fc3 because 09d2 is before 2fc3.

Conversions

OctetString provides three of its own conversion operations, none of which takes arguments:

FSS-UTF stands for File System Safe Unicode character set Transformation Format. It's a format described in Appendix F of The Unicode Standard, Version 1.1. FSS-UTF is designed to protect null bytes and ASCII slash characters used in various file systems--it encodes Unicode character strings into bytes that won't contain a null byte or an ASCII slash. Each Unicode character encoded in an FSS-UTF file may take one, two, or three bytes, so there's not the two-to-one correspondence between octets in an octet string and Unicode characters in a string that you might expect.

You will probably never have to worry about FSS-UTF specifics in Telescript programming because the operation asOctetString defined by the class String converts a Telescript string into an FSS-UTF octet string, and the operation asString converts the octet string back into the original string of Unicode characters.

The Stack Class Family

Stack is a class family that inherits from the class family List. Stack has two "percolating formals"--Item and Match--that have the same meaning they do in a list. That is, Item constrains the items that may be in the stack, and Match specifies how the stack determines two matching items.

Stack inherits all of List's features without specializing them. It also provides a set of five operations--pop, push, pushItems, roll, and swap--that make a Stack object a last-in, firsts-out (LIFO) stack. The top of the stack is position 1. The bottom of the stack is the highest position available (the length of the stack).

Initializing a Stack

You initialize a stack as you do a list: you first derive a class from the Stack class family by providing actuals for Item and Match; you then provide initialization arguments. The arguments become the items of the stack, arranged in the order in which you provided the arguments. The first initialization argument is at the top of the stack, the last argument is at the bottom.

Manipulating a Stack

Stack's operations allow you to manipulate a stack in all the traditional ways:

Figure 9-1 : The stack on the left is the original stack. Its top five items are rolled two positions forward in the upper right stack. Its top three items are rolled one position backward in the lower right stack.

Example Code

The following short Telescript program shows how you can work with bit strings, octet strings, and stacks.

do{
	/* Initialize a bit string and convert it. */

	bitValue: = BitString(%"10010101100111001", %0, %1);
	"The bit string after initialization:".dump();
	bitValue.dump();

	octetValue: = bitValue.asOctetString();
	"After converting the bit string to an octet string:".dump();
	octetValue.dump();

	"The octet string as an integer:".dump();
		octetValue.asInteger().dump();

	/* Initialize a stack and manipulate it. */

	filo: = Stack[Integer, Equal](46, 37, 25, 13, 2);
	"The stack after initialization:".dump();
	filo.dump();

	filo.pushItems(List[Integer, Equal](354, 265, 100));
	"The stack after pushing a list of three items:".dump();
	filo.dump();

	filo.roll(2,5);
	"The stack after rolling the top 5 items 2 places:".dump();
	filo.dump();
}
When you execute the program, you get the following results. Note that dump lists the items of a stack from bottom to top so a stack printout looks inverted. You have to turn your perspective on its head to see what's going on with the stacks.

String: <The bit string after initialization:>
BitString: <1001010110011100101>
String: <After converting the bit string to an octet string:>
OctetString: <959C>
String: <The octet string as an integer:>
Integer: <38300>
String: <The stack after initialization:>
Stack: <5 elements>
        Integer: <2>
        Integer: <13>
        Integer: <25>
        Integer: <37>
        Integer: <46>
String: <The stack after pushing a list of three items:>
Stack: <8 elements>
        Integer: <2>
        Integer: <13>
        Integer: <25>
        Integer: <37>
        Integer: <46>
        Integer: <100>
        Integer: <265>
        Integer: <354>
String: <The stack after rolling the top 5 items 2 places:>
Stack: <8 elements>
        Integer: <2>
        Integer: <13>
        Integer: <25>
        Integer: <265>
        Integer: <354>
        Integer: <37>
        Integer: <46>
        Integer: <100>
The program initializes a bit string named bitValue, converts it to an octet value that it uses to create an octet string named octetValue, then converts the octet string to an integer value. Notice that the initialization arguments for the bit string contain both a bit-string literal and individual bits; they're concatenated to create a single bit string. The program then creates a five-item stack, pushes on a list of three more items, and rolls the top five items two places.

The Set Class Family

Set, like List, is one of the two built-in immediate subclass families of Collection. Set brings the characteristic of uniqueness to the unordered items of a collection--it checks its items to make sure that no one item matches any other item in the set.

Set's features are designed to avoid duplicate items by checking any new item added to the set against existing items. If the set finds a match, it excludes the existing item from the set before including the new item.

The uniqueness of items in a set isn't always guaranteed. It can be compromised when an item is changed by an external action--that is, an action that does not take place through the features of the set. For example, an object outside the set may have an independent reference to an item in the set. That outside object may change the value of the item without the knowledge of the set and may--by chance or design--change the set item so that it matches another set item. If that happens, the set is said to be internally inconsistent. A set whose items are all unique is said to be internally consistent. You can, as you'll see, guard against internal inconsistency.

The Set class family has two formals that percolate up to Collection: Item and Match. These formals have the same meaning that they do in the Collection class family, and are used to derive a Collection superclass.

Set also has one mix-in superclass: Verified. This class offers an operation--verify--that checks a set's integrity and reports if the set is internally consistent. You can then take actions to fix the integrity or recreate the set.

Initializing a Set

To create a Set object, you must first provide actuals for the formals Item and Match to derive a class. The Item actual sets the class type of items that may be included in the set; the Match actual determines how two items are considered a match. Keep in mind, when setting Match, that if you supply Equal as the match, it creates a much more restrictive set than Same. That's because a set that only considers two items that are the same object as a match doesn't check for copy-equal items. A set defined with Equal will exclude copy-equal items.

Once you've supplied actuals, you supply a variable number of arguments for initialization just as you do when initializing a collection--the arguments are a group of objects that become the items of the set. When initializing a set, as when initializing a collection, the order of the object you supply doesn't determine the order of the items in the set because the items are unordered.

There is one important difference between initializing a set and initializing a collection: when you initialize a set, the set ensures that no two of its items match. When a set is initialized, the initialize operation checks each object as it's passed in to become an item in the set. If an object matches an item already in the set, the set item is discarded to make way for the new item. This ensures that a newly initialized set has no matching items--that it's internally consistent. And for this reason, the order of the initialization arguments can make a difference in rare cases.

Using Set Interactions

A set offers three operations that you can use to make two sets interact. Each operation accepts a set as an argument; the contents of the argument set aren't affected:

Note that all three of these operations have undefined behavior if either of the two sets they use is internally inconsistent--so you can't count on these operations to work with sets that have internally matching items.

Assuring Set Integrity

Set's mix-in superclass Verified adds the operation verify to a set. The operation accepts no arguments. It's specialized so that when you call it, it returns the boolean true only if the set is internally consistent--in other words, it has no matching items. Use this operation as necessary to confirm the integrity of a set.

Example Code

This short program creates four sets and uses them to interact with each other.

do{
	/* Initialize four sets and work with them. */

	oddSet: = Set[Integer, Equal](3, 5, 7, 9);
	oddShortSet: = Set[Integer, Equal](1, 3, 5, 7);
	evenSet: = Set[Integer, Equal](4, 6, 8);
	mixedSet: = Set[Integer, Equal](1, 2, 3);

	oddSet.union(evenSet);
	"oddSet (3,5,7,9) after union with (4,6,8):".dump();
	oddSet.dump();

	oddSet.intersection(oddShortSet);
	"oddSet after intersection with (1,3,5,7):".dump();
	oddSet.dump();

	oddSet.difference(mixedSet);
	"oddSet after difference with (1,2,3):".dump();
	oddSet.dump();
}
When you execute the program, you get the following results:

String: <oddSet (3,5,7,9) after union with (4,6,8):>
Set: <7 elements>
        Integer: <3>
        Integer: <5>
        Integer: <7>
        Integer: <9>
        Integer: <4>
        Integer: <6>
        Integer: <8>
String: <oddSet after intersection with (1,3,5,7):>
Set: <3 elements>
        Integer: <3>
        Integer: <5>
        Integer: <7>
String: <oddSet after difference with (1,2,3):>
Set: <2 elements>
        Integer: <5>
        Integer: <7>
The program creates four sets. It uses the set oddSet as the responder, using its operations to create a union, an intersection, and a difference with each of the other three sets in turn. Notice that oddSet itself changes after each operation, so the operations have a cumulative effect on oddSet.

The Dictionary Class Family

The Dictionary class family is a subclass of the Set class family. It's used primarily to store objects for indexed retrieval. To make this possible, a dictionary uses key-value pairs. Each item of the set is a key, which is an object used to add or request a stored object. Each key is associated with a single value, which is an object stored for retrieval. And each key-value pair stored in a dictionary is an entry.

You should think of a dictionary as a set of key-value pairs (not just items). Whenever you modify a dictionary, you work with key-value pairs. When you use the inherited Collection operations include or exclude, you include or exclude a key-value pair. And whenever you use the inherited Set operations difference, union, and intersection, you add or delete key-value pairs.

It's important to realize that only the keys of a dictionary are considered items, so only the keys are kept unique within the dictionary. The values paired with the keys do not have to be unique. Whenever you include a new key-value pair, the dictionary checks only the key against other keys to enforce the dictionary's internal consistency.

Initializing a Dictionary

To create a Dictionary object, you must first provide actuals for its three formals: Key, Value, and Match. The actual you supply for Key sets the constraint for objects that can be used as keys within the dictionary. The actual you supply for Value sets the constraint for objects that can be used as values within the dictionary. And the actual you supply for Match sets the way a match is determined within the dictionary. (It works the same as Match does for Collection.)

Key and Match correspond to the formals Item and Match that you find in Set and are, in fact, percolating formals. The actuals you supply for Key and Match are applied to Item and Match in Set to derive a superclass for the dictionary.

As initialization arguments for a dictionary, you supply key-value pairs. This means that you must supply an even number of arguments. Each odd-numbered argument (1, 3, 5, 7, and so on) is a key; each even-numbered argument (2, 4, 6, 8, and so on) is a value. A key is associated with the value that immediately follows to create a key-value pair: argument 1 with argument 2, argument 3 with argument 4, and so on.

When the dictionary is initialized, the key-value pairs are included into the dictionary one at a time starting at the beginning of the arguments. The key of each new pair is checked against the keys of existing pairs. If it matches, the existing pair it matches is excluded from the dictionary before the new pair is included.

Modifying a Dictionary

You can modify a dictionary using inherited operations. The Set operations difference, intersection, and union are specialized to handle key-value pairs. When you use these operations, any keys you include in or exclude from a dictionary have their values included or excluded with them. The Collection operations include, exclude, and clear are likewise specialized to include or exclude a full key-value pair anytime they include or exclude a key--even though include and exclude accept only a single argument (a key). If include adds a key to a dictionary, a value of nil is provided as the key's corresponding value. (Unless, of course, the value is constrained so it won't accept a nil, in which case the operation throws an exception.)

Dictionary also provides its own modifying operations:

You can use Dictionary's set operation with the same special syntax that's available for setting a list's item. The syntax is an assignment expression that takes this form:

<object>[<key>] = <value>
This expression, for example,

tempi["allegro"] = 120
sets a dictionary entry whose key is "allegro" and whose value is 120. The expression is equivalent to this feature expression:

tempi.set("allegro", 120)

Examining a Dictionary

Dictionary provides operations you can use to examine a dictionary without modifying its contents:

You can also use Dictionary's get operation with special syntax--an expression that takes this form:

<object>[<key>]
The expression evaluates to the value associated with the specified key. This expression, for example,

tempi["allegro"]
evaluates to the value associated with "allegro"(which might be 120 if you used the assignment in the last example). The expression is equivalent to this feature expression:

tempi.get("allegro")

Example Code

This program initializes a dictionary, modifies it, then retrieves a value and a key from the dictionary.

1	do{
2		/* Initialize a dictionary, modify it, and examine it. */

3		lookUp: = Dictionary[Character, String, Equal]
4				('R', "Red", 'G', "Green", 'B', "Blue");
5		"The dictionary after initialization:".dump();
6		lookUp.dump();

7		lookUp.add('C', "Cyan");
8		"The dictionary after adding an entry:".dump();
9		lookUp.dump();

10		lookUp['B'] = "Violet";
11		lookUp['M'] = "Magenta";
12		"The dictionary after setting B and M entries:".dump();
13		lookUp.dump();

14		lookUp.rekey('B', 'V');
15		"The dictionary after rekeying B to V:".dump();
16		lookUp.dump();

17		"The value associated with G:".dump();
18		lookUp['G'].dump();

19		"The key associated with Violet:".dump();
20		lookUp.find("Violet").dump();
21	}
When you execute the program, you get the following results:

String: <The dictionary after initialization:>
Dictionary: <3 elements>
        Character: <82(R)> String: <Red>
        Character: <71(G)> String: <Green>
        Character: <66(B)> String: <Blue>
String: <The dictionary after adding an entry:>
Dictionary: <4 elements>
        Character: <82(R)> String: <Red>
        Character: <71(G)> String: <Green>
        Character: <66(B)> String: <Blue>
        Character: <67(C)> String: <Cyan>
String: <The dictionary after setting B and M entries:>
Dictionary: <5 elements>
        Character: <82(R)> String: <Red>
        Character: <71(G)> String: <Green>
        Character: <66(B)> String: <Violet>
        Character: <67(C)> String: <Cyan>
        Character: <77(M)> String: <Magenta>
String: <The dictionary after rekeying B to V:>
Dictionary: <5 elements>
        Character: <82(R)> String: <Red>
        Character: <71(G)> String: <Green>
        Character: <67(C)> String: <Cyan>
        Character: <77(M)> String: <Magenta>
        Character: <86(V)> String: <Violet>
String: <The value associated with G:>
String: <Green>
String: <The key associated with Violet:>
Character: <86(V)>
The program creates a dictionary and initializes it with three entries. Notice that the odd-numbered arguments in the initialization statement are the keys, the even-numbered arguments are the values. The program adds an entry, then it sets two entries in lines 10 and 11. Notice that line 10, which sets an entry using a key that already exists, replaces the existing entry B, Blue with B, Violet. Line 11, which sets an entry using a key that doesn't exist, simply adds a new entry to the dictionary.

* * *

This brings us to the end of our look at collection classes--with the exception of String, which we'll examine in detail in the next chapter.

9 - Working With Collections
An Overview of Collection Classes
The Collection Classes
Classes Associated With Collection Classes
Class Families and Inheritance
Inheriting From a Derived Class
Inheriting From a Class Family
An Example
The Collection Class Family
Specifying the Type of Collection Items
Specifying the Way a Match Is Determined
The Compared Mix-In
The Equal Mix-In
The Same Mix-In
Initializing a Collection
Modifying a Collection
Examining a Collection
Using an Iterator
Copying and Converting a Collection
Example Code
The List Class Family
Initializing a List
Modifying a List
Specifying a Position
Working With Single Items
Working With Lists and Sublists
Examining a List
Comparing Lists
Example Code
The List Subclasses
The BitString Class
Bit-String Literals
Initializing a Bit String
Ordering Bit Strings
Conversion
The OctetString Class
Octet-String Literals
Initializing an Octet String
Ordering Octet Strings
Conversions
The Stack Class Family
Initializing a Stack
Manipulating a Stack
Example Code
The Set Class Family
Initializing a Set
Using Set Interactions
Assuring Set Integrity
Example Code
The Dictionary Class Family
Initializing a Dictionary
Modifying a Dictionary
Examining a Dictionary
Example Code

TS Ref - 26 JUN 1996

Generated with Harlequin WebMaker