Is it a shuffle?

Wheat Wizard 01/02/2018. 11 answers, 2.056 views
code-golf decision-problem combinatorics card-games

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.

Task

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
5 Comments
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?

11 Answers


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.

Haskell, 41 bytes

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.

Haskell, 43 bytes

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, 75 60 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./

Try it here! or Verify all the test cases.

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.
2 comments
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!


HighResolutionMusic.com - Download Hi-Res Songs

1 The Chainsmokers

Beach House flac

The Chainsmokers. 2018. Writer: Andrew Taggart.
2 (G)I-DLE

POP/STARS flac

(G)I-DLE. 2018. Writer: Riot Music Team;Harloe.
3 Ariana Grande

​Thank U, Next flac

Ariana Grande. 2018. Writer: Crazy Mike;Scootie;Victoria Monét;Tayla Parx;TBHits;Ariana Grande.
4 Anne-Marie

Rewrite The Stars flac

Anne-Marie. 2018. Writer: Benj Pasek;Justin Paul.
5 Clean Bandit

Baby flac

Clean Bandit. 2018. Writer: Jack Patterson;Kamille;Jason Evigan;Matthew Knott;Marina;Luis Fonsi.
6 Nicki Minaj

No Candle No Light flac

Nicki Minaj. 2018. Writer: Denisia “Blu June” Andrews;Kathryn Ostenberg;Brittany "Chi" Coney;Brian Lee;TJ Routon;Tushar Apte;ZAYN;Nicki Minaj.
7 BlackPink

Kiss And Make Up flac

BlackPink. 2018. Writer: Soke;Kny Factory;Billboard;Chelcee Grimes;Teddy Park;Marc Vincent;Dua Lipa.
8 Imagine Dragons

Bad Liar flac

Imagine Dragons. 2018. Writer: Jorgen Odegard;Daniel Platzman;Ben McKee;Wayne Sermon;Aja Volkman;Dan Reynolds.
9 BTS

Waste It On Me flac

BTS. 2018. Writer: Steve Aoki;Jeff Halavacs;Ryan Ogren;Michael Gazzo;Nate Cyphert;Sean Foreman;RM.
10 Halsey

Without Me flac

Halsey. 2018. Writer: Halsey;Delacey;Louis Bell;Amy Allen;Justin Timberlake;Timbaland;Scott Storch.
11 Little Mix

Woman Like Me flac

Little Mix. 2018. Writer: Nicki Minaj;Steve Mac;Ed Sheeran;Jess Glynne.
12 Brooks

Limbo flac

Brooks. 2018.
13 Fitz And The Tantrums

HandClap flac

Fitz And The Tantrums. 2017. Writer: Fitz And The Tantrums;Eric Frederic;Sam Hollander.
14 Backstreet Boys

Chances flac

Backstreet Boys. 2018.
15 Lady Gaga

I'll Never Love Again flac

Lady Gaga. 2018. Writer: Benjamin Rice;Lady Gaga.
16 Diplo

Close To Me flac

Diplo. 2018. Writer: Ellie Goulding;Savan Kotecha;Peter Svensson;Ilya;Swae Lee;Diplo.
17 Rita Ora

Velvet Rope flac

Rita Ora. 2018.
18 Bradley Cooper

Always Remember Us This Way flac

Bradley Cooper. 2018. Writer: Lady Gaga;Dave Cobb.
19 Imagine Dragons

Machine flac

Imagine Dragons. 2018. Writer: Wayne Sermon;Daniel Platzman;Dan Reynolds;Ben McKee;Alex Da Kid.
20 Erika Sirola

Speechless flac

Erika Sirola. 2018. Writer: Teemu Brunila;Stefan Dabruck;Jürgen Dohr;Guido Kramer;Dennis Bierbrodt;Chris Braide;Robin Schulz.

Related questions

Hot questions

Language

Popular Tags