Strona główna » Algorytmy » Artykuły » Efektywne Wyszukiwanie Anagramów
 

Efektywne Wyszukiwanie Anagramów

· część 1 ·

Zadanie

Dana jest lista wyrazów. Zadanie polega na efektywnym pogrupowaniu tak, że w każdej grupie znajdą się anagramy. Zakładamy, że wielkość liter nie ma znaczenia tj. A i a są tą samą literą. Wypisz następnie znalezione grupy. Przetestuj działania napisanego programu.

Analiza Zadania

Podczas rozwiązywania tego zadania należy rozwiązać dwie trudności. Pierwsza z nich dotyczy szybkości porównywania dwóch wyrazów czy są anagramami oraz szybkie grupowania wyrazów. Ten drugi problem może zostać rozwiązany poprzez zastosowanie słownika. Wtedy jest jednak potrzebny unikalny klucz, który będzie wspólny dla anagramów.

Klucz można utworzyć na podstawie wyrazu poprzez posortowanie jego liter. Nie jest to jednak metoda zbyt optymalna, ponieważ wymaga co najmniej O(n·log(n)) operacji. W anagramach kolejność liter nie ma znaczenia, więc przypiszmy każdemu znakowi alfabetu wartość tj. a = 1, b = 2, .., z = 26. Hash dla wyrazu będzie iloczynem podanych wartości dla liter w nim występujących. Przykładowo 'abc' będzie miało wartość 6. Tak samo jak 'acb' czy 'cba'.

Na podstawie obliczonego hasha algorytm będzie wstawiał odpowiednie dane do słownika. Na koniec wystarczy wypisać wartości przypisane do kolejnych kluczy jako oddzielne grupy.

Efektywność Algorytmu

Przyjmując, że na liście znajduje się n wyrazów, a wyszukanie pozycji do jego wstawienia wymaga log(n) porównań (wykorzystując wyszukiwanie binarne) to końcowa złożoność wynosi O(nlog(n)). Ograniczeniem może się okazać pojemność słownika - jeśli każdy wyraz będzie mieć inny hash to pomieszczenie wszystkich razem będzie niemożliwe.

Limity Implementacji

Jak długie wyrazy będzie można grupować? Otóż jeśli przyjmiemy, że długość wyrazu to k to będzie 26k możliwych hashy. Dla takiej wartości potrzebna liczba bitów to: n ≥ log2(26k) = k·log226. Oto tabelka, która określa jak długie wyraz można umieścić w zależności od wybranego rozmiaru danych:

Ile bitówDługość wyrazu
163
326
6413
12827

W powyższej tabeli zakładamy, że wyraz składa się z 26 liter alfabetu łacińskiego.

Implementacja

Hash

Funkcja Policz() ma za zadanie obliczyć hash dla podanego wyrazu.

  1. long Policz(string slowo) {
  2.   long wynik = 1;
  3.   for (int i = 0; i < slowo.size(); i++) {
  4.     if (isalpha(slowo[i])) {
  5.       wynik *= (tolower(slowo[i]) - 'a' + 1);
  6.     }
  7.   }
  8.   return wynik;
  9. }

Funkcja składa się z pętli, która przechodzi po kolejnych znakach. Kiedy wykryje literę mnoży aktualny wynik przez jej pozycję w alfabecie. Znak jest zamieniany na mały znak, ponieważ znaki mogą mieć różną wielkość.

Hash

Funkcja Grupuj() na podstawie podanych wyrazów grupuje je w słowniku. Elementy są grupowane po obliczonym hashu funkcją Policz().

  1. Slownik Grupuj(vector<string> &wyrazy) {
  2.   Slownik slownik;
  3.   for each(string wyraz in wyrazy) {
  4.     long hash = Policz(wyraz);
  5.     Slownik::iterator it;
  6.     it = slownik.find(hash);
  7.     if (it != slownik.end()) {
  8.       it->second.push_back(wyraz);
  9.     } else {
  10.       vector<string> lista;
  11.       lista.push_back(wyraz);
  12.       slownik.emplace(hash, lista);
  13.     }
  14.   }
  15.   return slownik;
  16. }

Dla każdego wyrazu obliczany jest hash. Jeśli obliczona wartość nie występuje jako klucz to należy dopisać nowy. W przeciwnym wypadku wystarczy dopisać wyraz na koniec listy pod danym kluczem.

Testowanie funkcji

Do przetestowania napisanej funkcji można skorzystać z poniższego kodu. Po zwróceniu wyniku program wypisuje znalezione grupy.

  1. int main () {
  2.   vector<string> dane;
  3.   dane.push_back("bca");
  4.   dane.push_back("abc");
  5.   dane.push_back("dca");
  6.   dane.push_back("dcb");
  7.   Slownik wynik = Grupuj(dane);
  8.   Slownik::iterator it = wynik.begin();
  9.   int grupa = 0;
  10.   while(it != wynik.end()) {
  11.     cout << "Grupa " << (grupa++) << ": ";
  12.     for each(string wyraz in it->second) {
  13.       cout << wyraz << " ";
  14.     }
  15.     cout << endl;
  16.     it++;
  17.   }
  18.   system("pause");
  19.   return 0;
  20. }