a design exercise
Posted: Fri Jul 31, 2020 2:07 am
The Problem
Design a representation of complex numbers and a function to do complex multiplication which satisfy the following:
“Structural” (⍴, ↑, etc.), “selection” (⌿, [;], ⌷, etc.), and the ⍳ family (⍳, ∊) functions work best on complex arrays if the complex number representation specifies a scalar representation. Therefore, probably, we want a complex number to be a scalar rather than (say) a 2-element vector.
An Attempt
For example: A complex number is an enclosed pair of numbers, and times is defined as follows:
So far so good.
To get a complex number from the enclosed pairs representation and vice versa, proceed as follows. The function “j” is described in the APL Chat Forum post j. on 2020-04-09.
Does the function f satisfy the conformability requirement?
OK, the above works. But it fails the second requirement for reduction with complex multiplication:
Discussion
I ran into this problem with the work on Rational Arithmetic reported in the APL Chat Forum. The problem is not a trivial one because complications engendered thereby propagate into all code using rational numbers. I am using complex numbers here to make the point with less distractions. I believe this is a problem with the Dyalog/NARS/APL2 style of arrays, reduction, scan, and inner product; I further believe that this problem precludes an APL programmer being able to define abstract scalars (complex numbers, rational numbers, extended precision integers, polynomials, etc.) in a general way. The problem was first described in Orth’s paper A Comparison of the IPSA and STSC Implementations of Operators and General Arrays, APL Quote Quad, volume 12, number 2, 1981-12.
There are several requirements for an abstract scalar to behave similarly as a simple scalar:
a f b g c (where g is another scalar function).
A Solution
The stated problem actually has a solution. Not general and not elegant, but a solution. To wit:
The solution is not very satisfying because basically the code is detecting whether it is being evaluated in a reduction, times/⍵, or by itself, ⍺ times ⍵.
I feel I am ready to update the Rational Arithmetic code to use these ideas, and then to use that code in some applications. Writing the applications and running into trouble was what made me realize things were amiss with the original design.
Design a representation of complex numbers and a function to do complex multiplication which satisfy the following:
- ⍺ times ⍵ works on corresponding subarrays for “conformable” complex arrays ⍺ and ⍵
- times⌿⍵ or times/⍵ works on a complex “vector” ⍵
“Structural” (⍴, ↑, etc.), “selection” (⌿, [;], ⌷, etc.), and the ⍳ family (⍳, ∊) functions work best on complex arrays if the complex number representation specifies a scalar representation. Therefore, probably, we want a complex number to be a scalar rather than (say) a 2-element vector.
An Attempt
For example: A complex number is an enclosed pair of numbers, and times is defined as follows:
f0← {(-/⍺×⍵),+/⍺×⌽⍵}
f ← f0¨
(⊂3 4) f (⊂5 6)
┌─────┐
│¯9 38│
└─────┘
3j4 × 5j6
¯9J38
So far so good.
To get a complex number from the enclosed pairs representation and vice versa, proceed as follows. The function “j” is described in the APL Chat Forum post j. on 2020-04-09.
j←{⍺←0 ⋄ ⍺+0j1×⍵}
jfp← j/¨
pfj← 9 11∘○¨
pfj 3j4 5j6
┌───┬───┐
│3 4│5 6│
└───┴───┘
jfp (3 4) (5 6)
3J4 5J6
Does the function f satisfy the conformability requirement?
y←(5 6) (7 8) (¯3 1)
(⊂3 4) f y
┌─────┬──────┬──────┐
│¯9 38│¯11 52│¯13 ¯9│
└─────┴──────┴──────┘
jfp (⊂3 4) f y
¯9J38 ¯11J52 ¯13J¯9
3j4 × 5j6 7j8 ¯3j1
¯9J38 ¯11J52 ¯13J¯9
xr←¯10+?4 5⍴22 ⋄ xi←¯10+?4 5⍴22 ⋄ x←xr j xi ⋄ x2←xr,¨xi
yr←¯10+?4 5⍴22 ⋄ yi←¯10+?4 5⍴22 ⋄ y←yr j yi ⋄ y2←yr,¨yi
(x×y) ≡ jfp x2 f y2
1
OK, the above works. But it fails the second requirement for reduction with complex multiplication:
y←(5 6) (7 8) (¯3 1)
f⌿ y
┌─────────────┐
│┌──────┬────┐│
││0 ¯210│0 96││
│└──────┴────┘│
└─────────────┘
f/y ⍝ It is not a matter of / versus ⌿
┌─────────────┐
│┌──────┬────┐│
││0 ¯210│0 96││
│└──────┴────┘│
└─────────────┘
×/ 5j6 7j8 ¯3j1
¯43J¯259
Discussion
I ran into this problem with the work on Rational Arithmetic reported in the APL Chat Forum. The problem is not a trivial one because complications engendered thereby propagate into all code using rational numbers. I am using complex numbers here to make the point with less distractions. I believe this is a problem with the Dyalog/NARS/APL2 style of arrays, reduction, scan, and inner product; I further believe that this problem precludes an APL programmer being able to define abstract scalars (complex numbers, rational numbers, extended precision integers, polynomials, etc.) in a general way. The problem was first described in Orth’s paper A Comparison of the IPSA and STSC Implementations of Operators and General Arrays, APL Quote Quad, volume 12, number 2, 1981-12.
There are several requirements for an abstract scalar to behave similarly as a simple scalar:
- If f is a “scalar function”, then ⍺ f ⍵ should work on conformable arrays ⍺ and ⍵, producing an array with the same type.
- If f is a “scalar function”, then f/⍵ on a vector ⍵ should produce a scalar with the same type.
- ⍺ ⍵ ↔ ⍺,⍵: The strand of two scalars juxtaposed should form a 2-element vector of the same type.
a f b g c (where g is another scalar function).
A Solution
The stated problem actually has a solution. Not general and not elegant, but a solution. To wit:
times0← {(-/⍺×⍵),+/⍺×⌽⍵} ⍝ note: same as f0 above
times← {2=≡⍵:⍺ times0¨⍵ ⋄ ⍺ times0 ⍵}
y←(5 6) (7 8) (¯3 1)
(⊂3 4) times y
┌─────┬──────┬──────┐
│¯9 38│¯11 52│¯13 ¯9│
└─────┴──────┴──────┘
3j4 × 5j6 7j8 ¯3j1
¯9J38 ¯11J52 ¯13J¯9
times/ y
┌────────┐
│¯43 ¯259│
└────────┘
×/ 5j6 7j8 ¯3j1
¯43J¯259
The solution is not very satisfying because basically the code is detecting whether it is being evaluated in a reduction, times/⍵, or by itself, ⍺ times ⍵.
xr←¯10+?4 5⍴22 ⋄ xi←¯10+?4 5⍴22 ⋄ x←xr j xi ⋄ x2←xr,¨xi
yr←¯10+?4 5⍴22 ⋄ yi←¯10+?4 5⍴22 ⋄ y←yr j yi ⋄ y2←yr,¨yi
(x×y) ≡ jfp x2 times y2
1
(×\y) ≡ jfp times\ y2
1
(×⍀y) ≡ jfp times⍀ y2
1
yr←¯10+?5 6⍴22 ⋄ yi←¯10+?5 6⍴22 ⋄ y←yr j yi ⋄ y2←yr,¨yi
plus←{⍺+⍵}
(x+.×y) ≡ jfp x2 plus.times y2
1
(x∘.×y) ≡ jfp x2 ∘.times y2
1
I feel I am ready to update the Rational Arithmetic code to use these ideas, and then to use that code in some applications. Writing the applications and running into trouble was what made me realize things were amiss with the original design.