c0=: _10 _12 _14 _15 _16 _18
c1=: 121 169 289 361
m=: 0 1 0 1 0 0 0 1 0 1
(/:/:m){c0,c1
_10 121 _12 169 _14 _15 _16 289 _18 361
(c0,c1)/:/:m
_10 121 _12 169 _14 _15 _16 289 _18 361
On Mon, Jan 28, 2013 at 8:49 AM, Simon Barker <[email protected]>wrote:
> (/:/:m){c0,c1
>
> Based on an old APL idiom.
>
> -----Original Message-----
> From: [email protected]
> [mailto:[email protected]] On Behalf Of Zsbán
> Ambrus
> Sent: 28 January 2013 16:37
> To: Programming forum
> Subject: [Jprogramming] Conditional by filtering then merging an array
>
> This question is about evaulating a conditional expression on each element
> of a list.
>
> Let's take a simple working example. Let's say we have this input,
>
> ]a =: 10+i.10
> 10 11 12 13 14 15 16 17 18 19
>
> and we want to invoke one of two different operations on its elements
> according to this mask:
>
> ]m =: 1 p: a
> 0 1 0 1 0 0 0 1 0 1
>
> We can of course do the operations individually, like this:
>
> m (-@])`(*:@])@.["0 a
> _10 121 _12 169 _14 _15 _16 289 _18 361
>
> Or we can compute both operations on the whole list then merge them like
> this:
>
> ]b =: m} (-a) ,: *:a
> _10 121 _12 169 _14 _15 _16 289 _18 361
>
> Octave (at least a recent enough version) does support this latter method
> as
> well:
>
> octave> a = 10:19,
> a =
> 10 11 12 13 14 15 16 17 18 19
> octave> m = isprime(a),
> m =
> 0 1 0 1 0 0 0 1 0 1
> octave> b = ifelse(m, a.^2, -a),
> b =
> -10 121 -12 169 -14 -15 -16 289 -18 361
>
> But it turns out the octavers have a different idiom for conditional
> expressions, which works by breaking the list to two parts according to the
> mask, invoke a list function on each part, then merge the two lists.
>
> octave> c1 = a(m).^2, c0 = -a(~m), c = []; c(m) = c1; c(~m) = c0,
> c1 =
> 121 169 289 361
> c0 =
> -10 -12 -14 -15 -16 -18
> c =
> -10 121 -12 169 -14 -15 -16 289 -18 361
>
> Now breaking the list to two parts and running a list function on each part
> is easy enough in J:
>
> ]c0 =: -(-.m)#a
> _10 _12 _14 _15 _16 _18
> ]c1 =: *:m#a
> 121 169 289 361
>
> But how do you merge the two resulting lists? There are some ugly ways,
> such as these
>
> ]c =: (I.-.m) C.^:_1 c1,c0
> _10 121 _12 169 _14 _15 _16 289 _18 361
> (m#^:_1 c1) + ((-.m)#^:_1 c0) NB. works only for numeric lists
> _10 121 _12 169 _14 _15 _16 289 _18 361
> m} ((-.m)#^:_1 c0) ,: (m#^:_1 c1)
> _10 121 _12 169 _14 _15 _16 289 _18 361
> c0 (I.-.m)} m#^:_1 c1
> _10 121 _12 169 _14 _15 _16 289 _18 361
>
> but is there a nicer way to merge two lists according to a mask like this?
> The mask tells which of the two lists each element of the result comes from
> but with the inputs containing only as many elements as necessary?
>
> Ambrus
> ----------------------------------------------------------------------
> For information about J forums see http://www.jsoftware.com/forums.htm
>
> ----------------------------------------------------------------------
> For information about J forums see http://www.jsoftware.com/forums.htm
>
----------------------------------------------------------------------
For information about J forums see http://www.jsoftware.com/forums.htm