Давши дерево генерувати свій Прюферський код


10

В код прюферово є унікальною послідовністю цілих чисел , яке позначає конкретне дерево.

Ви можете знайти код дерева Прюфера за допомогою наступного алгоритму, взятого з Вікіпедії:

Розглянемо мічене дерево T з вершинами {1, 2, ..., n}. На етапі i видаліть аркуш з найменшою міткою і встановіть i- й елемент послідовності Prüfer таким, що є міткою сусіда цього листа.

(Зауважте, що оскільки це листок, у нього буде лише один сусід).

Ви повинні зупинити ітерацію, коли на графіку залишаються лише дві вершини.

Завдання

Враховуючи позначене дерево як вхідний висновок, його код Prüfer. Ви можете брати участь у будь-який розумний спосіб. Наприклад, матриця суміжності або ваші мови, побудовані в графічному поданні. ( Ви не можете приймати дані як код Prüfer ).

Це є тому ви повинні прагнути мінімізувати байти у своєму джерелі.

Тестові справи

Ось деякі входи в ASCII з їх результатами нижче. Вам не потрібно підтримувати вхід ASCII таким чином.

    3
    |
1---2---4---6
    |
    5

{2,2,2,4}

1---4---3
    |
5---2---6---7
|
8

{4,4,2,6,2,5}

5---1---4   6
    |       |
    2---7---3

{1,1,2,7,3}

Чи можемо ми взяти вкорінене дерево як вхідне?
xnor

Чи можемо ми сприйняти вклад як щось подібне [[2,1],[2,3],[2,5],[2,4,6]]до першого випадку? (тобто кожна гілка)
HyperNeutrino

@xnor Так, ви можете
Ad Hoc Garf Hunter

1
Мені здається, що введення даних із ребрами або контурами, спрямованими на корінь, - це попередня обчислення коду Прюфера. Так чи інакше, я вважаю, що вам повинно бути зрозуміліше "Ви можете брати вклад будь-яким розумним чином (ви не можете приймати дані як код Прюфера)."
xnor

@xnor О, я не зрозумів, про що питає Гіпер Нейтрино.
Ad Hoc Hunter Hunter

Відповіді:


9

Математика, 34 байти

<<Combinatorica`
LabeledTreeToCode

Хтось повинен був це зробити….

Після завантаження Combinatoricaпакета функція LabeledTreeToCodeочікує введення дерева у вигляді непрямого графа з явно переліченими ребрами і вершинами; наприклад, вхід у другому тестовому випадку може бути Graph[{{{1, 4}}, {{4, 3}}, {{4, 2}}, {{2, 5}}, {{2, 6}}, {{6, 7}}, {{5, 8}}}, {1, 2, 3, 4, 5, 6, 7, 8}].


5
Звичайно, для цього є вбудований. > _>
HyperNeutrino

4

Пітон 3, 136 131 127 байт

def f(t):
 while len(t)>2:
  m=min(x for x in t if len(t[x])<2);yield t[m][0];del t[m]
  for x in t:m in t[x]and t[x].remove(m)

Приймає дані як матрицю суміжності. Перший приклад:

>>> [*f({1:[2],2:[1,3,4,5],3:[2],4:[2,6],5:[2],6:[4]})]
[2, 2, 2, 4]

ну я не зміг ...
HyperNeutrino

@HyperNeutrino Ви були приблизно на 4 секунди швидшими!
L3viathan

Хе-х-ю! І приблизно в 2,7 рази довше! : D gg
HyperNeutrino

1
delіснує? > _>
HyperNeutrino

1
@WheatWizard Ви маєте рацію щодо крапки з комою, але змішування вкладок і пробілів - помилка в Python 3.
L3viathan

2

Желе , 31 байт

FĠLÞḢḢ
0ịµÇHĊṙ@µÇCịṪ,
WÇÐĿḢ€ṖṖḊ

Монадічне посилання, яке приймає список пар вузлів (визначаючи ребра) у будь-якому порядку (і кожен у будь-якій орієнтації) і повертає Код Прюфера як список.

Спробуйте в Інтернеті!

Як?

FĠLÞḢḢ - Link 1, find leaf location: list of edges (node pairs)
F      - flatten
 Ġ     - group indices by value (sorted smallest to largest by value)
  LÞ   - sort by length (stable sort, so equal lengths remain in prior order)
    ḢḢ - head head (get the first of the first group. If there are leaves this yields
       -   the index of the smallest leaf in the flattened version of the list of edges)

0ịµÇHĊṙ@µÇCịṪ, - Link 2, separate smallest leaf: list with last item a list of edges
0ị             - item at index zero - the list of edges
  µ            - monadic chain separation (call that g)
   Ç           - call last link (1) as a monad (index of smallest leaf if flattened)
    H          - halve
     Ċ         - ceiling (round up)
      ṙ@       - rotate g left by that amount (places the edge to remove at the right)
        µ      - monadic chain separation (call that h)
         Ç     - call last link (1) as a monad (again)
          C    - complement (1-x)
            Ṫ  - tail h (removes and yields the edge)
           ị   - index into, 1-based and modular (gets the other node of the edge)
             , - pair with the modified h
               -    (i.e. [otherNode, restOfTree], ready for the next iteration)

WÇÐĿḢ€ṖṖḊ - Main link: list of edges (node pairs)
W         - wrap in a list (this is so the first iteration works)
  ÐĿ      - loop and collect intermediate results until no more change:
 Ç        -   call last link (2) as a monad
    Ḣ€    - head €ach (get the otherNodes, although the original tree is also collected)
      ṖṖ  - discard the two last results (they are excess to requirements)
        Ḋ - discard the first result (the tree, leaving just the Prüfer Code)

1

05AB1E , 29 байт

[Dg#ÐD˜{γé¬`U\X.å©Ï`XK`ˆ®_Ï]¯

Спробуйте в Інтернеті!

Пояснення

[Dg#                           # loop until only 1 link (2 vertices) remain
    ÐD                         # quadruple the current list of links
      ˜{                       # flatten and sort values
        γé                     # group by value and order by length of runs
          ¬`U                  # store the smallest leaf in X
             \X                # discard the sorted list and push X
               .å©             # check each link in the list if X is in that link
                  Ï`           # keep only that link
                    XK`ˆ       # add the value that isn't X to the global list
                        ®_Ï    # remove the handled link from the list of links
                           ]   # end loop
                            ¯  # output global list

1

Clojure, 111 байт

#(loop[r[]G %](if-let[i(first(sort(remove(set(vals G))(keys G))))](recur(conj r(G i))(dissoc G i))(butlast r)))

Необхідно, щоб вхід був хеш-картою, маючи мітки "як листочки" як ключі та "коренеподібні" мітки як значення. Наприклад:

{1 2, 3 2, 5 2, 4 2, 6 4}
{1 4, 3 4, 4 2, 8 5, 5 2, 7 6, 6 2}

На кожній ітерації він знаходить найменший ключ, на який не посилається жоден інший вузол, додає його до результату rта видаляє вузол із визначення графіка G. if-letпереходить в інший випадок, коли Gвін порожній, як firstповертається nil. Також слід скинути останній елемент.


Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.