Current issue

Vol.26 No.4

Vol.26 No.4

Volumes

© 1984-2024
British APL Association
All rights reserved.

Archive articles posted online on request: ask the archivist.

archive/22/1

Volume 22, No.1

J-ottings 45: A bit of OUTReach (or do I mean outrEACH?)

J-ottings 45: A bit of OUTReach (or do I mean outrEACH?)

by Norman Thomson

In APL2 there was just one primitive symbol (double dot) for “each” – the “pepper” effect is well imprinted on some of us with long memories, although, as Stephen Taylor hinted in his last editorial, APL just doesn’t look like that any more! I find it interesting that in spite of J being considerably richer in primitives than APL, it does not include “each” as a primitive. Nevertheless it is customary for many users to define an adverb “each”, even although it requires more characters to type than the J which defines it! Unfortunately, depending on the writer, the definition is sometimes &> , sometimes &.>, and occasionally "0 . I do not recommend this last practice, since a verb-noun combination is effectively one major case of a primitive operation, and in my view is best recognised as such. I therefore confine discussion of adverbial “each” to the other two possibilities, defined as

   each=.&>         NB. uses bond
   EACH=.&.>        NB. uses compose  

The definitions of the conjunctions u&v and u&.v in mathematical terms as (uv) and (v-1uv) show that the rule

   (f each) -: (>f EACH)

is completely general, and so only one of each and EACH is logically necessary. EACH has the general merit of greater compactness through generating what APL2 users would recognise as ragged arrays. On the other hand each has the merit of avoiding, or at least reducing, boxing on display, but often at the expense of requiring greater numbers of fill characters to achieve global homogeneity. In writing J it can be very desirable to be have a ‘rule of thumb’ appreciation of how each and EACH work without having to consider the semantic details of “box” and “open” at every point of use.

Thinking in terms of lists is helpful in using each and EACH effectively, as is sensitivity to the ways in which argument rank works. For example, given

   h=.i.2 3        NB. two 3-lists, rank = 2

compare the following

   h,each 6 7      NB. result=one 2-list, items=three 2-lists, rank=3
0 6
1 6
2 6

3 7
4 7
5 7

   h,EACH 6 7      NB. result=two 3-lists, items=2-lists, rank=2
+---+---+---+
|0 6|1 6|2 6|
+---+---+---+
|3 7|4 7|5 7|
+---+---+---+

Applying the general rule above (h,each 6 7) is identical to >h,EACH 6 7, or to put it in another way, the two displays above show the same six 2-lists, in one case individually boxed, in the other integrated into a structure of higher rank.

Using either EACH or each with a dyadic verb often requires the user to box one of the arguments to achieve a desired result through bringing about correct argument matching. Often this consists of a simple boxing of one of the arguments as in

   x=.6 7;8;9 10 11 12
   (<h),EACH x     NB. left rank 0, right rank 1, result rank 1
+-----+-----+----------+
|0 1 2|0 1 2|0  1  2  0|
|3 4 5|3 4 5|3  4  5  0|
|6 7 0|8 8 8|9 10 11 12|
+-----+-----+----------+  

The result is a 3-list of scalars (and so of rank 1) each made from the same scalar (<h) joined to each item of a 3-list, with everything opened and boxed following the (v-1uv) rule which defines &. (under). There is a local fill character in the first box, a local scalar expansion in the second, and the third containing two fill characters in the final column has a different shape ($) from the other two. Compare this with

   (<h),each x    NB. left rank 0, right rank 1, result rank 3
0   1   2   0 
3   4   5   0
6   7   0   0 

0   1   2   0
3   4   5   0  
8   8   8   0

0   1   2   0
3   4   5   0
9  10  11  12     

where there is no v-1 boxing. Again the result is a 3-list, but now not of scalars but of three filled lists, each of which is a 4-list of scalars after filling. APL2 talked about “ragged” and “heterogeneous” arrays – a more appropriate differentiation in J is between locally- and globally-filled lists, corresponding to EACH and each and respectively. Using "0 as an alternative gives a result rank between those of EACH and each :

  (<h),"0 x      NB. left rank 0, right rank 1, result rank 2
+-----+----------+
|0 1 2|6 7       |
|3 4 5|          |
+-----+----------+
|0 1 2|8         |
|3 4 5|          |
+-----+----------+
|0 1 2|9 10 11 12|
|3 4 5|          |
+-----+----------+ 

Here the scalar left argument gives rise to scalar replication, and the result is a 3-list of non-homogeneous 2-lists without fill characters, that is, it is only the display which is rectangular not the result object itself, as is made explicit by

   $EACH (<h),"0 x 
+---+-+ 
|2 3|2|
+---+-+ 
|2 3| |
+---+-+ 
|2 3|4|
+---+-+

In order to avoid monotonous repetition, subsequent examples will use just one of EACH or each, the other case being covered by the general rule (f each) -: (>f EACH) .Consideration of rank is almost always a preliminary to well-considered usage of EACH. There are no overall rank rules because these depend on the semantics of each individual f . In the next set of examples three figures in square brackets in a comment give the left, right and result ranks of the expression being commented. Consider first the basic case of the verb “shift” with a rank 1 argument on the left and a rank 2 argument on the right :

   0 1|.h           NB. 0-shift on rows ,: 1-shift on cols     [1 2 2]
1 2 0
4 5 3  

Applying EACH with “box” on either right or left changes the ranks :

   0 1 |.EACH<h     NB. 0-shift on h ; 1-shift on h            [1 0 1]
+-----+-----+
|0 1 2|3 4 5|
|3 4 5|0 1 2|
+-----+-----+

   (<0 1)|.EACH h   NB. (<0 1)-shift applied to items of h     [0 2 2]
+-+-+-+
|0|1|2|
+-+-+-+
|3|4|5|
+-+-+-+  

Between these extremes is the possibly more useful case of “left rank 1, right rank 1”:

   0 1|.EACH<"1 h   NB. 0-shift to 1st row ; 1-shift to 2nd     [1 1 1] 
+-----+-----+
|0 1 2|4 5 3|
+-----+-----+

Next; given the laminated rank 3 object :

   ]hlam=.h,:h+10   NB. hlam is h laminated with h+10
 0  1  2
 3  4  5
10 11 12
13 14 15  

here are the effects of four rather similar expressions, along with annotated descriptions which attempt to explain the differences between them. What is the best medium for such descriptions? Why J, of course, hence the comments which follow each of the four executable lines supply matching equivalent statements

   (<0 1)|.EACH <"2 hlam     NB. (0 1|.h);(0 1|.h+10)          [0 1 1]
+-----+--------+ 
|1 2 0|11 12 10| 
|4 5 3|14 15 13|
+-----+--------+

   0 1|.EACH <"2 hlam        NB. (0|.h);(1|.h+10)              [1 1 1] 
+-----+--------+
|0 1 2|13 14 15| 
|3 4 5|10 11 12|
+-----+--------+

   0 1|.EACH <"1 hlam        NB. (0|.EACH<"1 h),:(1|.EACH<"1 h+10) 
+--------+--------+ 
|0 1 2   |3 4 5   |          NB.                               [1 2 2]
+--------+--------+
|11 12 10|14 15 13|
+--------+--------+ 

This example is also equivalent to 2 2$0 0 1 1|.EACH ,<"1 hlam , since ,<"1 hlam is a 4-list whose items are 3-lists such as 0 1 2. The next example illustrates how "0 means apply a shift at the scalar level which is necessarily a “do nothing” operation.

   (0 1)|."0<"2 hlam         NB. equivalent to <"2 hlam        [1 1 1]
+-----+--------+
|0 1 2|10 11 12|
|3 4 5|13 14 15|
+-----+--------+  

A further set of examples illustrates what happens with characters. g is a pair of 2-lists and thus has the same shape as h, only its items are characters lists rather numbers.

   ]g=.2 3$'ant';'bee';'cat';'dog';'elk';'frog'
+---+---+----+
|ant|bee|cat |
+---+---+----+
|dog|elk|frog|
+---+---+----+  

Where items are characters lists, there are necessarily possible ambiguities between genuine space characters and space fill characters, something which is clarified in

   1 |.EACH g                NB. 1-shift at item level         [0 2 2]
 +---+---+----+
 |nta|eeb|atc |
 +---+---+----+
 |ogd|lke|rogf|
 +---+---+----+  

The next example shows simultaneous opening of both arguments followed by v-1 boxing :

   (<0 1)|.EACH <g           NB. <0 1|.g                       [0 0 0]
+--------------+
|+---+----+---+|
||bee|cat |ant||
|+---+----+---+|
||elk|frog|dog||
|+---+----+---+|
+--------------+      

The next example is one in which the result rank is less than an argument rank :

   0 1|.EACH <g              NB. (<0|.g),(<1|.g)               [1 0 0]
+--------------+--------------+
|+---+---+----+|+---+---+----+|
||ant|bee|cat |||dog|elk|frog||
|+---+---+----+|+---+---+----+|
||dog|elk|frog|||ant|bee|cat ||
|+---+---+----+|+---+---+----+|
+--------------+--------------+

In the final example in this set the shapes of the items within the two outer boxes are not identical, thus resolving an ambiguity between fill and genuine space characters which each would not have shown :

   0 1|.EACH<"1 g            NB. (<0|.0{g),(<1|.1{g)           [1 1 1]
+-------------+--------------+
|+---+---+---+|+---+----+---+|
||ant|bee|cat|||elk|frog|dog||
|+---+---+---+|+---+----+---+|
+-------------+--------------+  

However, given that each by definition reduces one level of boxing in every case, it often gives less fussy looking results than EACH where character lists are involved.

The examples so far have been chosen to illustrate the principles of each and EACH. An example of a practical problem involving each is that of evaluating linear expressions such as x + 2y - z over separate ranges of values of their variables. The results in this simple example are easy to check.

   t3=.1 2;3 4 5;6NB. x={1,2} y={3 4 5} z={6} 
   {t3                      NB. catalog generates all combinations
+-----+-----+-----+
|1 3 6|1 4 6|1 5 6|
+-----+-----+-----+
|2 3 6|2 4 6|2 5 6|
+-----+-----+-----+
 
   ip=.+/ .* each           NB. inner product each 
   ({t3)ip <1 2 _1          NB. 1 2 _1 are a set of linear coefficients
1 3 5
2 4 6  

The advent of “each” and ragged arrays in APL2 breathed new life into the language. J deals with such matters more subtly by forcing choices between two possibilities! Broadly speaking EACH makes objects more compact by allowing raggedness and reducing homogeneity to a local level, each reduces boxing at the cost of imposing greater homogeneity through globally applied fill characters.


script began 22:18:57
caching off
debug mode off
cache time 3600 sec
indmtime not found in cache
cached index is fresh
recompiling index.xml
index compiled in 0.2196 secs
read index
read issues/index.xml
identified 26 volumes, 101 issues
array (
  'id' => '10010590',
)
regenerated static HTML
article source is 'HTML'
source file encoding is 'ASCII'
read as 'Windows-1252'
URL: ../arch.css => trad/v221/../arch.css
completed in 0.241 secs