Шаги перестановки

10

Напишите функцию, которая принимает набор целых чисел и печатает каждую перестановку набора, а перестановка выполняется между каждым шагом.

вход

набор целых чисел, например (0, 1, 2)

Вывод

список перестановок и перестановок в формате (set) (swap) (set) ...

Прецедент

Input: 
(3, 1, 5)

Output:
(3, 1, 5)
(3, 1)
(1, 3, 5)
(3, 5)
(1, 5, 3)
(1, 3)
(3, 5, 1)
(3, 5)
(5, 3, 1)
(3, 1)
(5, 1, 3)

правила

  • Вы можете отформатировать набор чисел, как вы хотите.
  • Вы можете делать свопы в любом порядке
  • Вы можете повторить перестановки и перестановки, чтобы получить новый
  • Ваш код не должен фактически выполнять свопы, вывод должен просто показать, какой своп был сделан между вашим последним и текущим
  • Ваш код должен функционировать только для наборов с 2 или более элементами
  • В данном наборе не будет повторяющихся элементов (например, (0, 1, 1, 2) недопустимо)

Это код-гольф, поэтому выигрывает самый короткий код!

Billyoyo
источник
Можем ли мы использовать случайность?
Згарб
Вы имеете в виду просто делать случайные перестановки, пока не достигнете всех перестановок? Да, но вы должны быть уверены, что все перестановки были напечатаны
Billyoyo
3
Добро пожаловать в PPCG! Хороший первый вызов. Я бы предложил отредактировать пример, чтобы элементы не путались с индексами, такими как use set (3, 1, 4)или около того - читая его в первый раз, я сильно растерялся, потому что при первом обмене 0,1менялись не только элементы, 0,1но и индексы 0,1, а затем следующий своп не следовал этому образцу. Я также укажу вам на Песочницу, где вы можете публиковать вызовы и получать отзывы, прежде чем публиковать их на основном сайте.
AdmBorkBork
2
@TimmyD спасибо за предложение, я изменил пример. Я видел ссылку на песочницу сразу после того, как я разместил это, я буду публиковать там сначала с этого момента!
Billyoyo
1
-Джонсон-Троттер Штейнгауз алгоритм генерирует минимальную необходимую последовательность.
Нил

Ответы:

3

Mathematica, 102 байта

<<Combinatorica`
Riffle[#,BlockMap[Pick[#[[1]],#!=0&/@({1,-1}.#)]&,#,2,1]]&@*MinimumChangePermutations

Примеры

// столбец для более четкого результата

%[{1,3,5}]//Column
(*
{1,3,5}
{1,3}
{3,1,5}
{3,5}
{5,1,3}
{5,1}
{1,5,3}
{1,3}
{3,5,1}
{3,5}
{5,3,1}
*)
njpipeorgan
источник
3

Java, 449 426 байт

import java.util.*;interface P{static Set s=new HashSet();static void main(String[]a){o(Arrays.toString(a));while(s.size()<n(a.length)){p(a);o(Arrays.toString(a));}}static<T>void o(T z){System.out.println(z);s.add(z);}static int n(int x){return x==1?1:x*n(x-1);}static void p(String[]a){Random r=new Random();int l=a.length,j=r.nextInt(l),i=r.nextInt(l);String t=a[j];a[j]=a[i];a[i]=t;System.out.println("("+a[j]+","+t+")");}}

Метод грубой силы. Он продолжает делать случайные перестановки, пока не будут выполнены все возможные перестановки. Он использует набор строкового представления массива, чтобы проверить, сколько разных состояний было сгенерировано. Для n различных целых чисел n! = 1 * 2 * 3 * .. * n различных перестановок.

Обновить

  • Следовал советам Кевина Круйссена поиграть в гольф.

Ungolfed:

import java.util.*;

interface P {

    static Set<String> s = new HashSet<>();

    static void main(String[] a) {
        // prints the original input
        o(Arrays.toString(a));
        while (s.size() < n(a.length)) {
            p(a);
            // prints the array after the swap
            o(Arrays.toString(a));
        }
    }

    static void o(String z) {
        System.out.println(z);
        // adds the string representation of the array to the HashSet
        s.add(z);
    }

    // method that calculates n!
    static int n(int x) {
        if (x == 1) {
            return 1;
        }
        return x * n(x - 1);
    }

    // makes a random swap and prints what the swap is
    static void p(String[] a) {
        Random r = new Random();
        int l = a.length, j = r.nextInt(l), i = r.nextInt(l);
        String t = a[j];
        a[j] = a[i];
        a[i] = t;
        System.out.println("(" + a[j] + "," + t + ")");
    }
}

Применение:

$ javac P.java
$ java P 1 2 3
[1, 2, 3]
(2,1)
[2, 1, 3]
(1,1)
[2, 1, 3]
(2,2)
[2, 1, 3]
(3,1)
[2, 3, 1]
(3,1)
[2, 1, 3]
(1,2)
[1, 2, 3]
(1,1)
[1, 2, 3]
(3,2)
[1, 3, 2]
(2,3)
[1, 2, 3]
(3,1)
[3, 2, 1]
(3,1)
[1, 2, 3]
(3,3)
[1, 2, 3]
(1,2)
[2, 1, 3]
(1,3)
[2, 3, 1]
(1,2)
[1, 3, 2]
(3,1)
[3, 1, 2]

Как видите, свопов намного больше, чем необходимо. Но похоже на работу :-D

В качестве бонуса он работает и со строками, т.е.

$ java P 'one' 'two'
[one, two]
(two,one)
[two, one]
Master_ex
источник
у вас есть версия для игры в гольф, чтобы мы могли взглянуть на ваш метод?
Billyoyo
@Billyoyo: добавлен код без гольфа. Впрочем, ничего особенного :-)
Master_ex
Вы можете немного поиграть в гольф. Нет необходимости исправить предупреждения, так что вы можете удалили Set декларации: Set s=new HashSet();. Ваш код в методе nможет быть один возврат: static int n(int x){return x==1?1:x*n(x-1);}. И вы можете заменить String zв методе oс родовым вместо: static<T>void o(T z){System.out.println(z);s.add(z);}. Все вместе это уменьшит до 426 байтов .
Кевин Круйссен
1

JavaScript (ES6), 186 байт

f=
a=>{console.log(a);d=a.slice().fill(-1);s=[...d.keys()];for(i=l=a.length;i;)s[k=(j=s.indexOf(--i))+d[i]]<i?(console.log(a[s[j]=s[k]],a[s[k]=i]),console.log(s.map(n=>a[n])),i=l):d[i]*=-1}
;
<input id=i><input type=button value=Go! onclick=f(i.value.split`,`)>

Примечание: я не уверен, насколько гибкий формат вывода, может быть, я мог бы сделать это для 171 байта:

a=>{console.log(a);d=a.slice().fill(-1);s=[...d.keys()];for(i=l=a.length;i;)s[k=(j=s.indexOf(--i))+d[i]]<i?console.log(a[s[j]=s[k]],a[s[k]=i],s.map(n=>a[n],i=l)):d[i]*=-1}

Работает, выполняя алгоритм Steinhaus – Johnson – Trotter для массива индексов в случайном порядке и переводя обратно во входной массив. Ungolfed:

function steps(array) {
    console.log(array); // initial row
    var d = a.slice().fill(-1); // direction values
    var s = [...a.keys()]; // initial (identity) shuffle
    var l = a.length;
    for (var i = l; i; ) { // start by trying to move the last element
        var j = s.indexOf(--i);
        var k = j + d[i]; // proposed exchange
        if (s[k] < i) { // only exchange with lower index (within bounds)
            console.log(a[s[k]],a[i]); // show values being exchanged
            s[j] = s[k];
            s[k] = i; // do the exchange on the shuffle
            console.log(s.map(n=>a[n])); // show the shuffled array
            i = l; // start from the last element again
        } else {
            d[i] *= -1; // next time, try moving it the other way
        } // --i above causes previous element to be tried
    } // until no movable elements can be found
}
Нил
источник
1

Рубин, 86 байт

puts (2..(a=gets.scan(/\d+/).uniq).size).map{|i|a.permutation(i).map{|e|?(+e*", "+?)}}
cia_rana
источник
1

Haskell - 135 байт

p=permutations;f=filter
q(a:b:xs)=(\x->f(uncurry(/=)).zip x)a b:q(b:xs);q _=[]
l=head.f(all((==2).length).q).p.p
f=zip.l<*>map head.q.l

вывод:

> f [3,1,5]
[([3,1,5],(3,1)),([1,3,5],(3,5)),([1,5,3],(1,5)),([5,1,3],(1,3)),([5,3,1],(5,3))]

Я использую стандартную permutationsфункцию, которая не основана на свопах, поэтому я беру перестановки перестановок и нахожу одну, которая оказывается цепочкой свопов.

судейская шапочка
источник