Кодегольф постоянный

20

Задача состоит в том, чтобы написать codegolf для перманента матрицы .

Перманент n-by- nmatrix A= ( ai,j) определяется как

введите описание изображения здесь

Здесь S_nпредставляет множество всех перестановок [1, n].

Как пример (из вики):

введите описание изображения здесь

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

Входная матрица всегда квадратная и будет не более 6 на 6. Вам также нужно будет иметь возможность обрабатывать пустую матрицу, которая имеет перманент 1. Нет необходимости иметь возможность обрабатывать пустую матрицу (она вызывала слишком много проблемы).

Примеры

Входные данные:

[[ 0.36697048+0.02459455j,  0.81148991+0.75269667j,  0.62568185+0.95950937j],
 [ 0.67985923+0.11419187j,  0.50131790+0.13067928j,  0.10330161+0.83532727j],
 [ 0.71085747+0.86199765j,  0.68902048+0.50886302j,  0.52729463+0.5974208j ]]

Выход:

-1.7421952844303492+2.2476833142265793j

Входные данные:

[[ 0.83702504+0.05801749j,  0.03912260+0.25027115j,  0.95507961+0.59109069j],
 [ 0.07330546+0.8569899j ,  0.47845015+0.45077079j,  0.80317410+0.5820795j ],
 [ 0.38306447+0.76444045j,  0.54067092+0.90206306j,  0.40001631+0.43832931j]]

Выход:

-1.972117936608412+1.6081325306004794j

Входные данные:

 [[ 0.61164611+0.42958732j,  0.69306292+0.94856925j,
     0.43860930+0.04104116j,  0.92232338+0.32857505j,
     0.40964318+0.59225476j,  0.69109847+0.32620144j],
   [ 0.57851263+0.69458731j,  0.21746623+0.38778693j,
     0.83334638+0.25805241j,  0.64855830+0.36137045j,
     0.65890840+0.06557287j,  0.25411493+0.37812483j],
   [ 0.11114704+0.44631335j,  0.32068031+0.52023283j,
     0.43360984+0.87037973j,  0.42752697+0.75343656j,
     0.23848512+0.96334466j,  0.28165516+0.13257001j],
   [ 0.66386467+0.21002292j,  0.11781236+0.00967473j,
     0.75491373+0.44880959j,  0.66749636+0.90076845j,
     0.00939420+0.06484633j,  0.21316223+0.4538433j ],
   [ 0.40175631+0.89340763j,  0.26849809+0.82500173j,
     0.84124107+0.23030393j,  0.62689175+0.61870543j,
     0.92430209+0.11914288j,  0.90655023+0.63096257j],
   [ 0.85830178+0.16441943j,  0.91144755+0.49943801j,
     0.51010550+0.60590678j,  0.51439995+0.37354955j,
     0.79986742+0.87723514j,  0.43231194+0.54571625j]]

Выход:

-22.92354821347135-90.74278997288275j

Вы не можете использовать какие-либо ранее существующие функции для вычисления перманента.


источник
12
Не могли бы вы удалить сложное требование? Я думаю, что это разрушает в остальном хороший вызов. Каждый язык, в котором нет встроенной сложной арифметики, теперь должен выполнять совершенно отдельную задачу.
xnor
6
Если нам нужно обработать пустую матрицу, вы должны добавить ее в качестве контрольного примера. Тот факт, что вы не можете реально представить матрицу 0x0 списками, делает это немного сложным. Лично я бы просто убрал это требование.
Деннис
4
Нет смысла что-то класть в песочницу на 3 часа. Дайте ему 3 дня, и у людей будет возможность высказать свое мнение.
Питер Тейлор
7
1. Это не просто esolangs. Bash, например, не может даже иметь дело с поплавками . Исключение языка из соревнования только потому, что ему не хватает определенного числового типа, даже если он может легко реализовать желаемый алгоритм, просто придирчиво без уважительной причины. 2. Я все еще не уверен насчет пустой матрицы. Будет ли это [[]](имеет одну строку, пустая матрица отсутствует) или [](не имеет глубины 2, матрицы имеют) в форме списка?
Деннис
4
1. Я не думаю, что в Bash невозможно решить эту проблему, но если львиная доля кода используется для решения арифметики комплексных чисел, она перестает быть проблемой с перманентами. 2. Большинство, если не все текущие ответы - это языки без разбивки матрицы для ввода [[]].
Деннис

Ответы:

11

J, 5 байт

+/ .*

J не предлагает встроенные функции для перманента или определителя, но вместо этого предлагает конъюнкцию, u . v yкоторая рекурсивно расширяется yпо младшим и вычисляет двоичное число u . vмежду кофакторами и выходными данными рекурсивного вызова для младших. Выбор из uи vможет варьироваться. Например, при использовании , u =: -/и v =: *это , -/ .*который является определяющим фактором . Выбор может даже %/ .!где u=: %/, уменьшиться на деление, и v =: !который является биномиальным коэффициентом. Я не уверен, что означает этот вывод, но вы можете выбирать свои глаголы.

Альтернативная реализация для 47 байтов, использующая тот же метод в моем ответе Mathematica .

_1{[:($@]$[:+//.*/)/0,.;@(<@(,0#~<:)"+2^i.@#)"{

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

Другая реализация на 31 байт является

+/@({.*1$:\.|:@}.)`(0{,)@.(1=#)

которая является слегка гольфовой версией, основанной на расширении Лапласа, взятом из эссе о детерминантах .

использование

   f =: +/ .*
   f 0 0 $ 0 NB. the empty matrix, create a shape with dimensions 0 x 0
1
   f 0.36697048j0.02459455 0.81148991j0.75269667 0.62568185j0.95950937 , 0.67985923j0.11419187  0.50131790j0.13067928 0.10330161j0.83532727 ,: 0.71085747j0.86199765 0.68902048j0.50886302 0.52729463j0.5974208
_1.7422j2.24768
   f 0.83702504j0.05801749 0.03912260j0.25027115 0.95507961j0.59109069 , 0.07330546j0.8569899 0.47845015j0.45077079 0.80317410j0.5820795 ,: 0.38306447j0.76444045 0.54067092j0.90206306 0.40001631j0.43832931
_1.97212j1.60813
   f 0.61164611j0.42958732 0.69306292j0.94856925 0.4386093j0.04104116 0.92232338j0.32857505 0.40964318j0.59225476 0.69109847j0.32620144 , 0.57851263j0.69458731 0.21746623j0.38778693 0.83334638j0.25805241 0.6485583j0.36137045 0.6589084j0.06557287 0.25411493j0.37812483 , 0.11114704j0.44631335 0.32068031j0.52023283 0.43360984j0.87037973 0.42752697j0.75343656 0.23848512j0.96334466 0.28165516j0.13257001 , 0.66386467j0.21002292 0.11781236j0.00967473 0.75491373j0.44880959 0.66749636j0.90076845 0.0093942j0.06484633 0.21316223j0.4538433 , 0.40175631j0.89340763 0.26849809j0.82500173 0.84124107j0.23030393 0.62689175j0.61870543 0.92430209j0.11914288 0.90655023j0.63096257 ,: 0.85830178j0.16441943 0.91144755j0.49943801 0.5101055j0.60590678 0.51439995j0.37354955 0.79986742j0.87723514 0.43231194j0.54571625
_22.9235j_90.7428
миль
источник
1
Вау это все, что я могу сказать.
13

Haskell, 59 байт

a#((b:c):r)=b*p(a++map tail r)+(c:a)#r
_#_=0
p[]=1
p l=[]#l

Это делает подобную Лапласу разработку вдоль первого столбца и использует то, что порядок строк не имеет значения. Это работает для любого числового типа.

Ввод в виде списка списков:

Prelude> p [[1,2],[3,4]]
10
Кристиан Сиверс
источник
2
Всегда приветствуем решение Haskell!
8

Желе , 10 9 байт

Œ!ŒDḢ€P€S

Попробуйте онлайн!

Как это устроено

Œ!ŒDḢ€P€S  Main link. Argument: M (matrix / 2D array)

Œ!         Generate all permutations of M's rows.
  ŒD       Compute the permutations' diagonals, starting with the main diagonal.
    Ḣ€     Head each; extract the main diagonal of each permutation.
      P€   Product each; compute the products of the main diagonals.
        S  Compute the sum of the products.
Деннис
источник
Это просто слишком хорошо!
7

Python 2, 75 байт

Кажется неуклюжим ... должно быть побиваемо.

P=lambda m,i=0:sum([r[i]*P(m[:j]+m[j+1:],i+1)for j,r in enumerate(m)]or[1])
feersum
источник
6

05AB1E , 19 14 13 байт

œvyvyNè}Pˆ}¯O

Попробуйте онлайн!

объяснение

œ              # get all permutations of rows
 v        }    # for each permutation
  yv   }       # for each row in the permutation
    yNè        # get the element at index row-index
        P      # product of elements
         ˆ     # add product to global array
           ¯O  # sum the products from the global array
Emigna
источник
Слегка шокирующий размер ответа! Не могли бы вы дать какое-то объяснение?
@Lembik: кажется, что это может быть еще короче. У меня пока есть второе решение такого же размера.
Emigna
Обработка пустых матриц больше не требуется.
Деннис
8 байт с использованием карт . Жаль , что новый 05AB1E не поддерживает мнимые числа (или я просто не знаю , как), так как мы теперь имеем главную диагональ и это встроенная команда может быть 6 байт: œ€Å\PO.
Кевин Круйссен
5

Python 2, 139 байт

from itertools import*
def p(a):c=complex;r=range(len(a));return sum(reduce(c.__mul__,[a[j][p[j]]for j in r],c(1))for p in permutations(r))

repl.it

Реализует наивный алгоритм, который слепо следует определению.

Джонатан Аллан
источник
4

MATL, 17 14 байтов

tZyt:tY@X])!ps

Попробуйте онлайн

объяснение

t       % Implicitly grab input and duplicate
Zy      % Compute the size of the input. Yields [rows, columns]
t:      % Compute an array from [1...rows]
tY@     % Duplicate this array and compute all permutations (these are the columns)
X]      % Convert row/column to linear indices into the input matrix
)       % Index into the input matrix where each combination is a row
!p      % Take the product of each row
s       % Sum the result and implicitly display
Suever
источник
1
Очень впечатляюще.
4

Рубин, 74 63 байта

->a{p=0;a.permutation{|b|n=1;i=-1;a.map{n*=b[i+=1][i]};p+=n};p}

Простой перевод формулы. Несколько байтов сохранено благодаря ezrast.

объяснение

->a{
    # Initialize the permanent to 0
    p=0
    # For each permutation of a's rows...
    a.permutation{|b|
        # ... initialize the product to 1,
        n=1
        # initialize the index to -1; we'll use this to go down the main diagonal
        # (i starts at -1 because at each step, the first thing we do is increment i),
        i=-1
        # iteratively calculate the product,
        a.map{
            n*=b[i+=1][i]
        }
        # increase p by the main diagonal's product.
        p+=n
    }
    p
}
м-chrzan
источник
1
reduceна самом деле вредит вашему счету байтов по сравнению с агрегацией вручную:->a{m=0;a.permutation{|b|n=1;a.size.times{|i|n*=b[i][i]};m+=n};m}
ezrast
@ezrast Спасибо! Удалось также сыграть в гольф times.
m-chrzan
3

Ruby 2.4.0, 59 61 байт

Рекурсивное расширение Лапласа:

f=->a{a.pop&.map{|n|n*f[a.map{|r|r.rotate![0..-2]}]}&.sum||1}

Менее гольф:

f=->a{
  # Pop a row off of a
  a.pop&.map{ |n|
    # For each element of that row, multiply by the permanent of the minor
    n * f[a.map{ |r| r.rotate![0..-2]}]
  # Add all the results together
  }&.sum ||
  # Short circuit to 1 if we got passed an empty matrix
  1
}

Ruby 2.4 официально не выпущен. В более ранних версиях их .sumнужно будет заменить .reduce(:+), добавив 7 байтов.

ezrast
источник
2

Mathematica, 54 байта

Coefficient[Times@@(#.(v=x~Array~Length@#)),Times@@v]&

Теперь, когда пустые матрицы больше не рассматриваются, это решение является действительным. Это происходит от страницы MathWorld по перманентам .

миль
источник
@alephalpha Это хорошая идея использовать строки для определения коэффициентов, но разве это не сломается, если строки не будут уникальными?
миль
2

JavaScript (ES6), 82 байта

f=a=>a[0]?a.reduce((t,b,i)=>t+b[0]*f(a.filter((_,j)=>i-j).map(c=>c.slice(1))),0):1

Конечно, работает и с пустой матрицей.

Нил
источник
@ETHproductions я никогда не учусь ...
Нил
1
Точно мой код, только что опубликованный 14 часов назад, я попытаюсь добавить комплексные числа
edc65
2

Юлия 0,4 , 73 байта

f(a,r=1:size(a,1))=sum([prod([a[i,p[i]] for i=r]) for p=permutations(r)])

В более новых версиях julia вы можете отказаться []от понимания, но нужно using Combinatoricsдля permutationsфункции. Работает со всеми типами номеров в Юлии, в том числе Complex. rявляется UnitRangeобъектом, определенным как аргумент функции по умолчанию, который может зависеть от предыдущих аргументов функции.

Попробуйте онлайн!

GGGG
источник