Minimalizacja łączenia par polega na połączeniu wszystkich elementów tablicy przy jak najmniejszym koszcie. Zakładamy, że koszt połączenia pary to suma wartości łączonych liczb. Z kolei połączenie pary polega na zastąpieniu łączonych elementów ich sumą.
Weźmy przykładowo tablicę liczb całkowitych L:={3, 2, 5, 7}.
Lista | Łączone liczby | Koszt |
---|---|---|
{3, 2, 5, 7} | 2 i 3 | 5 |
{5, 5, 7} | 5 i 5 | 10 |
{10, 7} | 10 i 7 | 17 |
Sumując ostatnią kolumnę uzyskujemy całkowity koszt: 5 + 10 + 17 = 32. Jest to rozwiązanie optymalne, ponieważ uzyskaliśmy najmniejszy możliwy koszt.
Liczby możemy połączyć także nieoptymalnie. Oznacza to, że połączymy wszystkie pary, ale uzyskamy większy koszt:
Lista | Łączone liczby | Koszt |
---|---|---|
{3, 2, 5, 7} | 5 i 7 | 12 |
{3, 2, 12} | 3 i 12 | 15 |
{2, 15} | 2 i 15 | 17 |
Ostateczny koszt to 44. To więcej niż w pierwszym przykładzie.
Jeśli w danym kroku chcemy uzyskać jak najmniejszy koszt to musimy połączyć dwie najmniejsze liczby na liście. Załóżmy, że zawsze łączymy najmniejszą z liczbą równą lub większą. Wtedy jeśli mamy więcej niż jedno wystąpienie liczby z pary to wystarczy, że wybierzemy dowolną z nich.
Zastanówmy się teraz nad złożonością takiego algorytmu. Operacją dominującą jest tu porównanie. W i-tym kroku będziemy wykonywać n - i porównań, gdzie n to długość tablicy. Na podstawie tego można stwierdzić, że złożoność czasowa algorytmu będzie kwadratowa rzędu O(n2). Z kolei złożoność pamięciowa będzie O(1).
Przed przystąpieniem do implementacji trzeba określić dwie strategie. Pierwsza strategia dotyczy znalezienia dwóch najmniejszych liczb w ciągu. Przyjęta przeze mnie w rozwiązaniu strategia polega na zadeklarowaniu dwóch zmiennych, które będą przechowywać pozycje najmniejszych liczb. Przed rozpoczęciem porównań zakładam, że najmniejszy liczby to pierwsza i druga na liście. Dla każdego elementu prócz dwóch pierwszych: porównuje z pierwszą zmienną. Jeśli jest mniejsza to zapamiętuje tę pozycję. Jeśli jednak nie to porównuje z drugą dotychczas najmniejszą liczbą. Jeśli jest mniejsza to zastępuje drugą liczbę aktualnie rozpatrywaną. W ten sposób zawsze uzyskamy dwie najmniejsze liczby.
Druga strategia dotyczy zmniejszania listy. Zakładamy, że podaną listę możemy modyfikować. (Jeśli jednak chcemy tego uniknąć to wystarczy dodać nową funkcję nadrzędną lub na początku tej funkcji dodać kopiowanie listy.) Deklarowanie za każdym razem może okazać się czasochłonne, dlatego po znalezieniu najmniejszych liczb: sumę liczb (koszt) wstawimy w miejsce liczby najmniejszej znajdującej się bliżej początku. Z kolei liczbę na pozycji drugiej najmniejszej liczby zamienimy z ostatnim elementem na liście, a potem zmniejszymy długość listy o 1.
Przedstawiona poniżej funkcja wyliczKoszt() dla podanej tablicy zwraca minimalny koszt łączenia elementów.
(2.) Początkowo należy ustalić koszt na 0 i (3.) sprawdzić czy są pary do łączenia. Jeśli tak to (4.) przyjmujemy dwa pierwsze elementy za najmniejsze, a następnie (5. - 11.) weryfikujemy to przeglądając pozostałe elementy i aktualizując, które elementy są najmniejsze. Dla znalezionej pary: (12.) powiększ koszt, (13.) nową sumę umieść na liście zamiast pierwszej minimalnej wartości, a następnie (14.) przesuń drugą wartość na koniec tablicy i (15.) zmniejsz informację ile pozostało elementów do złączenia.
Napisane funkcje można przetestować przy pomocy poniższego kodu, który wczyta od użytkownika ile będzie liczb na liście n, a następnie n wartości. Po wczytaniu program wypisze minimalny koszt łączenia wczytanych liczb.
Napisz rekurencyjną wersję funkcji minimalizującą koszt łączenia par.