array theory question

View: New views
9 Messages — Rating Filter:   Alert me  

array theory question

by John Nowak :: Rate this Message:

Reply to Author | View Threaded | Show Only this Message

I know some of your here are familiar with APL, Nail, etc, so I  
thought I'd ask.

My question is this: Am I correct in thinking that single element  
arrays in Nial (and I believe also APL2) are equivalent to their  
contents? In other words, does {42} == 42? Also, does anyone know  
anything about the history of this question as it pertains to APL?  
Thanks in advance.

The rest of this email is confusing, but I left it in anyway. It may  
help explain why I'm interested in the question of {x} == x.

- - -

The reason I ask is that this has implications for the "constructive"  
language I recently mentioned. There seem to be five ways of passing N  
arguments to a function:

1. Use curried functions (not an option here)
2. Pass tuples when N > 1 (what I proposed previously, but has  
limitations)
3. Pass null-terminated lists when N > 1 (the approach of FP)
4. Pass null-terminated lists (stacks) in all cases (the concatenative  
approach)
5. Pass arrays in the style of Trenchard More's array theory (as in  
Nial, possibly APL2)

The first option (currying) is out because there's just no way I can  
see to combine such an approach with pointfree programming without  
making it horribly painful (like it is in Haskell).

The second approach (tuples) seems to have annoying limitations,  
although its simplicity is appealing.

The third approach (lists) is nicer than using tuples but it is still  
impossible to write a generic function that does partial application.  
The reason is that when partially applying to a function that takes  
two values, you need the resulting function to enlist the second  
argument passed to it before consing on the first. If the function  
takes three values however, you do not need to enlist because a list  
is already being passed in and you can just cons on the partially  
applied value and then apply as normal. I hope that makes sense...

The fourth approach is out because it makes the form of "construction"  
useless: If everything takes and returns a stack, then using  
construction will always result in a stack of stacks. For example,  
'[sq, sq] 5' would return '[[25][25]]'. Getting at the values  
afterwards is a huge pain.

The fifth approach has the benefits of the third but also allows  
partial application. If we decide that {x} == x, then we can have  
generic partial application because it will "automatically" enlist the  
object passed to the function resulting from the partial application  
if necessary. This sort of auto-enlisting also makes it much easier to  
generalize scalar operations to arrays (i.e. we can view a function  
like 'square' as one that operates on all elements of an array, but  
since the single element array is the same as the element it contains,  
we can also pass it scalar values).

Re: array theory question

by John Nowak :: Rate this Message:

Reply to Author | View Threaded | Show Only this Message


On Dec 22, 2008, at 6:32 PM, John Nowak wrote:

> My question is this: Am I correct in thinking that single element
> arrays in Nial (and I believe also APL2) are equivalent to their
> contents? In other words, does {42} == 42?

Actually, I think my question should've be more precise: Do single  
element arrays *containing atoms* equal the atoms that they contain?

I did manage Nial running. I somehow missed the binaries on the site  
before. It does indeed turn out that '42 = {42}' as I had understood  
from the array theory paper (http://www.nial.com/Documents.html).  
However, it is not the case that '{{1 2}} = {1 2}'.

I think this is exactly the behavior I need. All functions could take  
and return a single array, but construction (I guess called "atlas" in  
Nial) would still work nicely (e.g. '+ [1, 2]' == '[3]' == '3'). I  
suppose I'm reinventing something.

I think I've answered my question, but if anyone has thoughts on  
Nial's approach to arrays as opposed to, say, APL's or J's, I would  
very much appreciate them.

- John

Re: array theory question

by John Cowan :: Rate this Message:

Reply to Author | View Threaded | Show Only this Message

John Nowak scripsit:

> My question is this: Am I correct in thinking that single element  
> arrays in Nial (and I believe also APL2) are equivalent to their  
> contents? In other words, does {42} == 42? Also, does anyone know  
> anything about the history of this question as it pertains to APL?  
> Thanks in advance.

The answer to the question, as posed, is "No".  But a better question
is this: "Is a scalar an array"?  And the answer to that is "Yes, a
scalar is an array of rank 0, which by definition contains 1 element."
This is true AFAIK in all APL-derived languages.  A single-element array
can be of any rank.

Backing off to your underlying issue, there is no problem with either
identifying tuples of size 1 with scalars in languages that have tuples,
or not doing so.  ML, Haskell, and Pure do: Python and Q don't.

--
Work hard,                                      John Cowan
play hard,                                      cowan@...
die young,                                      http://www.ccil.org/~cowan
rot quickly.

Re: array theory question

by John Cowan :: Rate this Message:

Reply to Author | View Threaded | Show Only this Message

Scripsi:

> The answer to the question, as posed, is "No".  But a better question
> is this: "Is a scalar an array"?  And the answer to that is "Yes, a
> scalar is an array of rank 0, which by definition contains 1 element."
> This is true AFAIK in all APL-derived languages.  

Except, as it turns out, Q'Nial version 6 (the current version) which
no longer implements array theory exactly, according to Wikipedia.

--
Not to perambulate                 John Cowan <cowan@...>
    the corridors                  http://www.ccil.org/~cowan
during the hours of repose
    in the boots of ascension.       --Sign in Austrian ski-resort hotel

Re: array theory question

by John Nowak :: Rate this Message:

Reply to Author | View Threaded | Show Only this Message


On Dec 22, 2008, at 7:18 PM, John Cowan wrote:

> Backing off to your underlying issue, there is no problem with either
> identifying tuples of size 1 with scalars in languages that have  
> tuples,
> or not doing so.  ML, Haskell, and Pure do: Python and Q don't.

You are correct here. The issue is that I want to do it with lists/
arrays of arbitrary length, not tuples. In other words, I'd like this  
to work, where '&' is a psuedo partial application and ':' is  
application:

    (add2 & 2) : 3       ==>  5
    (add3 & 2) : {3, 4}  ==>  9

The question here is what the semantics of '&' are. In the former, it  
appears to be as such where '++' is 'cons':

    (f & x) : y  ==  f : (x ++ (y ++ {}))

In the latter however, the enlisting of the argument on the right side  
of the application is not necessary as it is already a list:

    (f & x) : {y, z}  ==  f : (x ++ {y, z})

What I'm hoping might make sense here is making the atom 'y'  
equivalent to '{y}'. This would allow the second form of '&' shown  
above to be used in the general case. Do you think that may be  
workable? Is it comparable to what Nial or APL do?

Thanks for the help.

- John


Re: array theory question

by John Cowan :: Rate this Message:

Reply to Author | View Threaded | Show Only this Message

John Nowak scripsit:

> What I'm hoping might make sense here is making the atom 'y'  
> equivalent to '{y}'. This would allow the second form of '&' shown  
> above to be used in the general case. Do you think that may be  
> workable? Is it comparable to what Nial or APL do?

Ah^2.  What you want is Algol 68's "rowing coercion" ("row" being A68
jargon for vector, or 1-D array).  This coercion transforms a non-row
value into the corresponding row value with one element.  It's applicable
only where the type is uniquely known at compile time: in A68 jargon,
in the following "strong contexts":

o the actual parameters of a function call
o a type cast
o the RHS of an assignment, contant declaration, or variable
        declaration with an initializer
o the body of a procedure (which is an expression in A68)
o all the arms of an if- or case-expression except one
o either side, but not both, of an identity expression

It's interesting note that A68, as of the Revised Report, has two kinds
of 2-D arrays:  "row row of <whatever>", which is a C-style rectangular
matrix, and "row of row of <whatever>", which is a Java-style vector of
pointers to vectors.  (These are the names of the types, not the way you
write them in declarations.)  Consequently, there are four kinds of 3-D
arrays, and in general 2^(k-1) kinds of rank-k arrays.

--
MEET US AT POINT ORANGE AT MIDNIGHT BRING YOUR DUCK OR PREPARE TO FACE WUGGUMS
John Cowan      cowan@...      http://www.ccil.org/~cowan

Re: array theory question

by John Nowak :: Rate this Message:

Reply to Author | View Threaded | Show Only this Message


On Dec 22, 2008, at 8:56 PM, John Cowan wrote:

> John Nowak scripsit:
>
>> What I'm hoping might make sense here is making the atom 'y'
>> equivalent to '{y}'. This would allow the second form of '&' shown
>> above to be used in the general case. Do you think that may be
>> workable? Is it comparable to what Nial or APL do?
>
> Ah^2.  What you want is Algol 68's "rowing coercion" ("row" being A68
> jargon for vector, or 1-D array).  This coercion transforms a non-row
> value into the corresponding row value with one element.

Very interesting! I suppose the question I'd need to answer is if it's  
sufficient to only go from scalar -> vector. I think the answer is  
"no" if functions like 'square' are meant to operate on vectors rather  
than scalars as then 'square 5' would return a one element vector and  
I'd be back to the same problem regarding construction. It would be  
sufficient however for functions like 'sum' that always return a  
scalar but can consume either a scalar or vector as input. For things  
like 'square', there's always map.

The choices I make here are going to end up depending on the  
complexity of the necessary type system more than anything else. It's  
quite likely that I won't be able to offer a map-like operation that  
works on the vector passed to a function. This is essentially the same  
state of things in a language like Factor where there's no way of  
applying a function to every element of the stack (without using with-
datastack and then replacing the main stack or something awful).  
Instead, you have stack and arrays as separate concepts. If I go this  
route, the rowing coercion will be sufficient. The only place I think  
I really need to use it is for partial application anyway. It seems a  
shame though to need to add this notion to the type system just for  
this one case, although perhaps there's a nice way of working it into  
the notation so it seems obvious.

Thanks again.

- John

Re: array theory question

by stevan apter :: Rate this Message:

Reply to Author | View Threaded | Show Only this Message

this was a hot topic in the 70s, during which time several
vendors proposed different extensions to the APL\360 type
system.  in boxed or grounded array systems, enclosure of an
atom (a "simple scalar" of depth 0) has depth 1.  in nested
or floating or ungrounded array systems, enclosure of an
atom is a no op.  Sharp APL and J are boxed systems, IBM's
APL2 and STSC's Nested Arrays are floating systems.  

the most immediate effect of grounded vs. ungrounded systems
is that in the latter, but not the former, a simple scalar
can be an element of a nested array.  e.g. where <x is the
enclosure of x

    (<1 2 3;4)

is possible in APL2, but not in Sharp APL, which requires

    (<1 2 3;<4)

K is grounded, but strictly speaking has lists and not arrays.
K can have

    (1 2 3;4)

also see trenchard more's work on array theory:

http://domino.research.ibm.com/tchjr/journalindex.nsf/0/415fa6b9fe085ed785256bfa0068419d?OpenDocument


----- Original Message -----
From: "John Nowak" <john@...>
To: "concatenative" <concatenative@...>
Sent: Monday, December 22, 2008 6:32 PM
Subject: [stack] array theory question


>I know some of your here are familiar with APL, Nail, etc, so I  
> thought I'd ask.
>
> My question is this: Am I correct in thinking that single element  
> arrays in Nial (and I believe also APL2) are equivalent to their  
> contents? In other words, does {42} == 42? Also, does anyone know  
> anything about the history of this question as it pertains to APL?  
> Thanks in advance.
>
> The rest of this email is confusing, but I left it in anyway. It may  
> help explain why I'm interested in the question of {x} == x.
>
> - - -
>
> The reason I ask is that this has implications for the "constructive"  
> language I recently mentioned. There seem to be five ways of passing N  
> arguments to a function:
>
> 1. Use curried functions (not an option here)
> 2. Pass tuples when N > 1 (what I proposed previously, but has  
> limitations)
> 3. Pass null-terminated lists when N > 1 (the approach of FP)
> 4. Pass null-terminated lists (stacks) in all cases (the concatenative  
> approach)
> 5. Pass arrays in the style of Trenchard More's array theory (as in  
> Nial, possibly APL2)
>
> The first option (currying) is out because there's just no way I can  
> see to combine such an approach with pointfree programming without  
> making it horribly painful (like it is in Haskell).
>
> The second approach (tuples) seems to have annoying limitations,  
> although its simplicity is appealing.
>
> The third approach (lists) is nicer than using tuples but it is still  
> impossible to write a generic function that does partial application.  
> The reason is that when partially applying to a function that takes  
> two values, you need the resulting function to enlist the second  
> argument passed to it before consing on the first. If the function  
> takes three values however, you do not need to enlist because a list  
> is already being passed in and you can just cons on the partially  
> applied value and then apply as normal. I hope that makes sense...
>
> The fourth approach is out because it makes the form of "construction"  
> useless: If everything takes and returns a stack, then using  
> construction will always result in a stack of stacks. For example,  
> '[sq, sq] 5' would return '[[25][25]]'. Getting at the values  
> afterwards is a huge pain.
>
> The fifth approach has the benefits of the third but also allows  
> partial application. If we decide that {x} == x, then we can have  
> generic partial application because it will "automatically" enlist the  
> object passed to the function resulting from the partial application  
> if necessary. This sort of auto-enlisting also makes it much easier to  
> generalize scalar operations to arrays (i.e. we can view a function  
> like 'square' as one that operates on all elements of an array, but  
> since the single element array is the same as the element it contains,  
> we can also pass it scalar values).
>

Re: array theory question

by John Cowan :: Rate this Message:

Reply to Author | View Threaded | Show Only this Message

John Nowak scripsit:

> > Ah^2.  What you want is Algol 68's "rowing coercion" ("row" being A68
> > jargon for vector, or 1-D array).  This coercion transforms a non-row
> > value into the corresponding row value with one element.
>
> Very interesting! I suppose the question I'd need to answer is if it's  
> sufficient to only go from scalar -> vector. I think the answer is  
> "no" if functions like 'square' are meant to operate on vectors rather  
> than scalars as then 'square 5' would return a one element vector and  
> I'd be back to the same problem regarding construction.

Right.  However, rowing can be applied more than once.  Note that in
Algol 68, rowing is not done to the arguments of operators, so that
vector + scalar and matrix + scalar can do the right thing rather
than rowing the scalar and probably throwing an array conformity error
at compile time.

--
All Norstrilians knew what laughter was:        John Cowan
it was "pleasurable corrigible malfunction".    cowan@...
        --Cordwainer Smith, Norstrilia