Комбинации, диспозиции и перестановки в PHP

Каков наиболее эффективный способ генерации всех комбинаций, расположений и перестановок массива в PHP?

4 ответа

Решение

Вот код для получения всех перестановок:

http://php.net/manual/en/function.shuffle.php

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

Вы можете использовать этот класс: http://pear.php.net/package/Math_Combinatorics

и используйте это как:

$combinatorics = new Math_Combinatorics;

$words_arr = array(
    'one'   => 'a',
    'two'   => 'b',
    'three' => 'c',
    'four'  => 'd',
    );

for ($i=count($words_arr)-1;$i>=1;$i--) {
    echo '<br><br>' . $i . ':<br>';
    $combinations_arr = $combinatorics->combinations($words_arr, $i);
    foreach ($combinations_arr as $combinations_arr_item) {
        echo implode(', ', $combinations_arr_item) . '<br>';
    }
}

Я хотел бы предложить мое решение CombinationGenerator, который генерирует комбинации элементов массива.

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

class CombinationsGenerator
{
    public function generate(array $list): \Generator
    {
        if (count($list) > 2) {
            for ($i = 0; $i < count($list); $i++) {
                $listCopy = $list;

                $entry = array_splice($listCopy, $i, 1);
                foreach ($this->generate($listCopy) as $combination) {
                    yield array_merge($entry, $combination);
                }
            }
        } elseif (count($list) > 0) {
            yield $list;

            if (count($list) > 1) {
                yield array_reverse($list);
            }
        }
    }
}

$generator = new \CombinationsGenerator();

foreach ($generator->generate(['A', 'B', 'C', 'D']) as $combination) {
    var_dump($combination);
}

Это в стиле PHP7, он использует\Generator потому что я считаю, что для этого есть веские причины.

/* Combinations */
function nCr($n, $r)
{
  if ($r > $n)
  {
    return NaN;
  }
  if (($n - $r) < $r)
  {
    return nCr($n, ($n - $r));
  }
  $return = 1;
  for ($i = 0; $i < $r; $i++)
  {
    $return *= ($n - $i) / ($i +1);
  }
  return $return;
}

/* Permutations */
function nPr($n, $r)
{
  if ($r > $n)
  {
    return NaN;
  }
  if ($r)
  {
    return $n * (nPr($n -1, $r -1));
  }
  else
  {
    return 1;
  }
}

Мне пришлось изменить ответ @hejdav так, чтобы он включал частичные комбинации, чтобы он полностью предоставил все результаты.

Я искал Интернет для этого решения и по состоянию на июнь 2019 года, я считаю, что это единственный общедоступный ответ (где угодно), который действительно перечисляет все возможные, не дублирующие возможности.

class CombinationsGenerator
{
    /**
     * Taken from https://stackru.com/a/39447347/430062.
     * 
     * @param array $list
     * @return \Generator
     */
    public function generate(array $list): \Generator
    {
        // Generate even partial combinations.
        $list = array_values($list);
        $listCount = count($list);
        for ($a = 0; $a < $listCount; ++$a) {
            yield [$list[$a]];
        }

        if ($listCount > 2) {
            for ($i = 0; $i < count($list); $i++) {
                $listCopy = $list;

                $entry = array_splice($listCopy, $i, 1);
                foreach ($this->generate($listCopy) as $combination) {
                    yield array_merge($entry, $combination);
                }
            }
        } elseif (count($list) > 0) {
            yield $list;

            if (count($list) > 1) {
                yield array_reverse($list);
            }
        }
    }
}
Другие вопросы по тегам