How to interleave two given vectors in APL

605 Views Asked by At

I'm trying to solve a problem using APL, for which I have two vectors v1 and v2, with relative length of at most +1, depending on the input. That means that ((≢v1)-(≢v2))∊¯1 0 1.

What would be the best way to interleave said vectors, so to create a third vector v3 such that v3=v1[0],v2[0],v1[1],v2[1],...?

(If it's relevant, I'm using Dyalog APL version 16.0)

6

There are 6 best solutions below

1
On BEST ANSWER

This should work in just about every APL.

(v0,v1)[⍋(⍳⍴v0),⍳⍴v1]

If you want to worry about either v0 or v1 being scalars, then

(v0,v1)[⍋(⍳⍴,v0),⍳⍴,v1]
0
On

If you don't mind getting a prototype fill element when the vectors have unequal length, then

Interleave←{,⍉↑⍵}

will do. Try it online!

Otherwise, you can interleave the matching parts, and then append the missing element(s — it works for length-differences greater than one too):

Interleave←{
    lengths←⌊/≢¨⍵
    main←,⍉↑lengths↑¨⍵
    tail←⊃,/lengths↓¨⍵
    main,tail
}

Try it online!

0
On

Using a Dyalog dfn:

zip ← {
    mix ← ,⍉↑ ⍺ ⍵
    mask ← ,⍉↑ 1⊣¨¨ ⍺ ⍵
    mask / mix
}

The idea here is to mix both arguments, then transpose the result and finally flatten it (mix).

Then, we apply the same on an array of 1s, corresponding to the length of the given arrays (mask), and use it as a mask, to filter the prototypes added by the mix primitive.

Note that this allows also for zipping arrays with a difference of length greater than one element.

Try it online!

0
On

Here's how I would solve the original question in APL2:

LEN←∊⌊/⍴¨V1 V2 
V3←∊((LEN↑V1),¨LEN↑V2),LEN↓¨V1 V2
1
On

As I do not know Dyalog APL, I answer in the old ISO APL from the 1970's years:
(v1,v2)[⍋((0.5×(⍴v1)<⍴v2)+⍳⍴v1),((0.5×(⍴v2)<⍴v1)+⍳⍴v2]

The first element will be the one of the longuest vector, if they have the same length, the first element is the first element of v1.

0
On

With vectors of same length use the inner product:

 1 2 3,.,40 50 60

┌──────────────┐
│1 40 2 50 3 60│
└──────────────┘

Building on top of that begets this dfn :

{r←(⍴⍺)⌊⍴⍵⋄(∊(r⍴⍺),.,r⍴⍵),(r↓⍺),r↓⍵}

alternatively, we can laminate (keeping the same overall logic):

{r←(⍴⍺)⌊⍴⍵⋄(,(r⍴⍺),⍪r⍴⍵),(r↓⍺),r↓⍵}