Is it a shuffle?

Wheat Wizard 01/02/2018. 11 answers, 2.056 views

Yesterday I asked this question about riffle shuffles. It seems that yesterdays question was a bit too hard so this question is a related but much easier task.

Today you are asked to determine if a permutation is in fact a riffle shuffle. Our definition of riffle shuffle is adapted from our last question:

The first part of the shuffle is the divide. In the divide partition the deck of cards in two. The two subsections must be continuous, mutually exclusive and exhaustive. In the real world want to make your partition as close to even as possible, however in this challenge this is not a consideration, all partitions including ones that are degenerate (one partition is empty) are of equal consideration.

After they have been partitioned, the cards are spliced together in such a way that cards maintain their relative order within the partition they are a member of. For example, if card A is before card B in the deck and cards A and B are in the same partition, card A must be before card B in the final result, even if the number of cards between them has increased. If A and B are in different partitions, they can be in any order, regardless of their starting order, in the final result.

Each riffle shuffle can then be viewed as a permutation of the original deck of cards. For example the permutation

1,2,3 -> 1,3,2

is a riffle shuffle. If you split the deck like so

1, 2 | 3

we see that every card in 1,3,2 has the same relative order to every other card in it's partition. 2 is still after 1.

On the other hand the following permutation is not a riffle shuffle.

1,2,3 -> 3,2,1

We can see this because for all the two (non-trivial) partitions

1, 2 | 3
1 | 2, 3 

there is a pair of cards that do not maintain their relative orderings. In the first partition 1 and 2 change their ordering, while in the second partition 2 and 3 change their ordering.

Given a permutation via any reasonable method, determine if it represents a valid riffle shuffle. You should output two distinct constant values one for "Yes, this is a riffle shuffle" and one for "No, this is not a riffle shuffle".

This is so answers will be scored in bytes with less bytes being better.

Test Cases

1,3,2 -> True
3,2,1 -> False
3,1,2,4 -> True
2,3,4,1 -> True
4,3,2,1 -> False
1,2,3,4,5 -> True
1,2,5,4,3 -> False
5,1,4,2,3 -> False
3,1,4,2,5 -> True
2,3,6,1,4,5 -> False
1 Mr. Xcoder 01/02/2018
Can the output be inconsistent, but truthy / falsy in our language? Like (Python, where, among the integers only 0 is falsy) 0 for falsy but any integer in [1, +∞) for truthy?
1 Wheat Wizard 01/02/2018
@Mr.Xcoder I don't like the truthy/falsy values because they are rather hard to define well. Answers should stick to the current rules.
Ørjan Johansen 01/02/2018
Suggested test case: [3,1,4,2,5].
9 Ørjan Johansen 01/02/2018
Sorry about this, but: [2,3,6,1,4,5].
1 Dennis♦ 01/02/2018
Can we take permutations of [0, ..., n-1] instead of [1, ..., n] as input?

Arnauld 01/03/2018.

JavaScript (ES6), 47 bytes

Takes input as an array of integers. Returns a boolean.

([x,...a],y)=>a.every(z=>z+~x?y?z==++y:y=z:++x)

Test cases

let f =

([x,...a],y)=>a.every(z=>z+~x?y?z==++y:y=z:++x)

;[
[1,3,2      ],  // -> True
[3,2,1      ],  // -> False
[3,1,2,4    ],  // -> True
[2,3,4,1    ],  // -> True
[4,3,2,1    ],  // -> False
[1,2,3,4,5  ],  // -> True
[1,2,5,4,3  ],  // -> False
[3,1,4,2,5  ],  // -> True
[2,3,6,1,4,5]   // -> False
]
.forEach(a => console.log(JSON.stringify(a) + ' -> ' + f(a)))

How?

The input array A is a valid riffle shuffle if it consists of at most two distinct interlaced sequences of consecutive integers.

The challenge rules specify that we're given a permutation of [1...N]. So we don't need to additionally check that the sorted union of these sequences actually results in such a range.

We use a counter x initialized to A[0] and a counter y initially undefined.

For each entry z in A, starting with the 2nd one:

• We check whether z is equal to either x+1 or y+1. If yes, we increment the corresponding counter.
• Else: if y is still undefined, we initialize it to z.
• Else: we make the test fail.

xnor 01/03/2018.

n%h=n+0^(n-h)^2
f l=elem(foldl(%)0\$l++l)l

Try it online!

Checks if the list concatenated to itself contains the numbers 0..n-1 in order as a subsequence.

Ørjan Johansen 01/02/2018.

s takes a list of integers as in the OP examples and returns a Bool.

s p=or[f(<x)p++f(>=x)p<[1..]|x<-p]
f=filter

Try it online!

How it works

• The list comprehension tries each element x of p in turn and checks if it can be the first element of the second partition of the shuffle. Then or returns True if any of the checks were True.
• The comprehension does this by partitioning (with filter) p into elements smaller and larger than (or equal to) x, concatenating, and checking if the resulting list is [1..length p], i.e. the elements in order.
• The check for whether the resulting list is [1..length p] is done by seeing if the result is strictly smaller than the infinite list [1..] == [1,2,3,etc.], which gives the same result for any permutation.

Dennis 01/03/2018.

Jelly, 13 6 bytes

ỤIṢḊRẠ

Try it online!

Alternate version, postdates challenge, 5 bytes

Ụ>ƝSỊ

Try it online!

How it works

ỤIṢḊRẠ  Main link. Argument: A (permutation of [1, ..., n])

Ụ       Grade up; sort the indices of A by their respective values.
For shuffles, the result is the concatenation of up to two increasing
sequences of indices.
I      Compute the forward differences.
In a shuffle, only one difference may be negative.
Ṣ     Sort the differences.
Ḋ    Dequeue; remove the first (smallest) difference.
R   Range; map each k to [1, ..., k].
This yields an empty array for non-positive values of k.
Ạ  All; check if all resulting ranges are non-empty.

xnor 01/03/2018.

Python 2, 39 bytes

l=input()
n=0
for x in l*2:n+=n==x
l[n]

Try it online!

Takes the list 0-indexed and outputs via exit code.

ovs 01/02/2018.

Python 2, 7560 47 bytes

thanks to Dennis for -13 bytes

x={0}
for v in input():x=x-{v-1}|{v}
len(x)<3>q

Try it online!

Output is via exit code.

G B 01/02/2018.

Ruby, 35 bytes

->l{l.any?{|a|l&[*1..a]|l==l.sort}}

Try it online!

How?

• l & [*1..a] | l applies an intersection and then an union: first get the elements of l that are <=a and then add the remaining elements of l without changing the order. If a is the number we are looking for, then this operation is the same as sorting l.

Οurous 01/02/2018.

Clean, 56 55 bytes

import StdEnv
?l=any(\e=sortBy(\a b=a<e&&b>=e)l<[1..])l

Try it online!

xnor 01/03/2018.

Pyth, 6 bytes

}SQy*2

Test suite

}SQy*2

*2  the (implicit) input list doubled, that is concatenated with a copy
y    all subsequences of it
}       contain an element equaling
SQ     the input list sorted 

Checks whether the doubled input list contains a sorted version of itself as a subsequence.

Mr. Xcoder 01/02/2018.

Pyth, 9 bytes

!t-.+xLQS

Test suite.

Saved 3 bytes thanks to isaacg.

Pyth, 14 bytes

}SQm.nS.Tcd2./

Outputs True and False for riffle-shuffle and non-riffle-shuffle respectively.

How?

}SQm.nS.Tcd2./ ~ Full program. Reads input from STDIN and outputs to STDOUT.

./ ~ Return all divisions of the input into disjoint substrings (partition).
m           ~ Map over the above using a variable d.
cd2   ~ Chop d into two-element lists.
.T      ~ Justified transpose, ignoring absences.
S        ~ Sort (lexicographically).
.n         ~ Deep-flatten.
}              ~ Check if the above contains...
SQ            ~ The sorted input.
isaacg 01/02/2018
Furthermore, <#0 can be replaced by - for 2 more bytes.
Mr. Xcoder 01/02/2018
@isaacg Oh yeah facepalm thanks. Edited. Edited.

Mr. Xcoder 01/03/2018.

Husk, 7 bytes

±§#OoṖD

Test suite!