Is it a shuffle?

19

1

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

Post Rock Garf Hunter

Posted 2018-01-02T15:48:33.933

Reputation: 55 382

1Can 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? – Mr. Xcoder – 2018-01-02T16:50:16.457

1@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. – Post Rock Garf Hunter – 2018-01-02T16:53:00.587

Suggested test case: [3,1,4,2,5]. – Ørjan Johansen – 2018-01-02T17:28:44.683

9Sorry about this, but: [2,3,6,1,4,5]. – Ørjan Johansen – 2018-01-02T18:14:58.930

1Can we take permutations of [0, ..., n-1] instead of [1, ..., n] as input? – Dennis – 2018-01-02T22:37:13.140

@Dennis Sure, I was just starting at 1 for consistency any reasonable form of input is fine by me. – Post Rock Garf Hunter – 2018-01-02T22:38:05.070

Answers

8

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.

Arnauld

Posted 2018-01-02T15:48:33.933

Reputation: 111 334

6

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.

xnor

Posted 2018-01-02T15:48:33.933

Reputation: 115 687

5

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.

Ørjan Johansen

Posted 2018-01-02T15:48:33.933

Reputation: 6 914

5

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.

Dennis

Posted 2018-01-02T15:48:33.933

Reputation: 196 637

5

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.

xnor

Posted 2018-01-02T15:48:33.933

Reputation: 115 687

4

Brachylog, 9 bytes

o~cĊ⟨⊆⊇⟩?

Try it online!

The predicate succeeds if the input represents a riffle shuffle and fails if it does not, meaning among other things that if the predicate is run as an entire program success will print true. and failure will print false.. Input is taken as a list of any sorts of items and interprets it as representing a permutation of itself sorted.

   Ċ         Some length-two list
 ~c          which concatenated
o            is the input sorted
    ⟨        satisfies the condition that its first element
     ⊆       is an ordered not-necessarily-contiguous sublist
        ?    of the input
      ⊇      which is an ordered superlist of
       ⟩     the list's second element.

I feel like there's something in the spirit of ⊆ᵐ which should work in the place of the four-byte "sandwich" construct ⟨⊆⊇⟩.

Unrelated String

Posted 2018-01-02T15:48:33.933

Reputation: 5 300

1I think you are the first person to use a sandwich in a PPCG answer (and it's a beautiful symmetric one :)) – Fatalize – 2019-03-06T15:20:07.633

3

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.

ovs

Posted 2018-01-02T15:48:33.933

Reputation: 21 408

2

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.

G B

Posted 2018-01-02T15:48:33.933

Reputation: 11 099

2

Clean, 56 55 bytes

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

Try it online!

Οurous

Posted 2018-01-02T15:48:33.933

Reputation: 7 916

2

Pyth, 5 bytes

}SQy+

Test suite

}SQy+

    +QQ  concatenated two copies of the (implicit) input
   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.

Thanks to Erik the Outgolfer for 1 byte taking better advantage of implicit input with +QQ rather than *2Q.

xnor

Posted 2018-01-02T15:48:33.933

Reputation: 115 687

5 bytes: }SQy+. It gets expanded to }SQy+QQ. – Erik the Outgolfer – 2018-01-07T18:43:50.143

@EriktheOutgolfer Nice one, thanks. – xnor – 2018-01-10T20:34:50.357

1

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.

Mr. Xcoder

Posted 2018-01-02T15:48:33.933

Reputation: 39 774

Furthermore, <#0 can be replaced by - for 2 more bytes. – isaacg – 2018-01-02T22:41:40.117

@isaacg Oh yeah facepalm thanks. Edited. Edited. – Mr. Xcoder – 2018-01-02T22:41:41.270

0

Husk, 7 bytes

±§#OoṖD

Test suite!

Mr. Xcoder

Posted 2018-01-02T15:48:33.933

Reputation: 39 774

0

Perl, 28 bytes

Includes +1 for a

Input on STDIN, 1 based

perl -aE '$.+=$_==$.for@F,@F;say$.>@F' <<< "3 1 2 4"

Uses xnor's algorithm

Ton Hospel

Posted 2018-01-02T15:48:33.933

Reputation: 14 114

0

C (gcc), 61 bytes

a,b,B;f(int*A){for(a=b=B=1;*A;A++)a-*A?B*=*A>b,b=*A:++a;b=B;}

Try it online!

Jonathan Frech

Posted 2018-01-02T15:48:33.933

Reputation: 6 681