Python - dziel i rządź
W podejściu dziel i zwyciężaj problem, z którym się borykasz, jest dzielony na mniejsze podproblemy, a następnie każdy problem jest rozwiązywany niezależnie. Gdy będziemy dalej dzielić podproblemy na jeszcze mniejsze podproblemy, możemy w końcu dojść do etapu, w którym dalszy podział nie będzie możliwy. Te „atomowe” najmniejsze możliwe podproblemy (ułamki) są rozwiązane. Rozwiązanie wszystkich podproblemów jest ostatecznie łączone w celu uzyskania rozwiązania pierwotnego problemu.
Ogólnie możemy to zrozumieć divide-and-conquer podejście w trzystopniowym procesie.
Dziel / łam
Ten krok obejmuje podzielenie problemu na mniejsze podproblemy. Podproblemy powinny stanowić część pierwotnego problemu. Na tym etapie zwykle przyjmuje się podejście rekurencyjne, aby podzielić problem, dopóki żaden problem podrzędny nie będzie dalej podzielny. Na tym etapie podproblemy nabierają charakteru atomowego, ale nadal stanowią część rzeczywistego problemu.
Podbij / Rozwiąż
Na tym etapie do rozwiązania jest wiele mniejszych podproblemów. Generalnie na tym poziomie problemy uważa się za „rozwiązane” samodzielnie.
Scal / Połącz
Gdy mniejsze podproblemy są rozwiązane, na tym etapie są one rekurencyjnie łączone, aż do sformułowania rozwiązania pierwotnego problemu. To algorytmiczne podejście działa rekurencyjnie, a kroki zdobywania i scalania działają tak blisko, że wydają się jednością.
Przykłady
Poniższy program jest przykładem divide-and-conquer podejście programistyczne, w którym wyszukiwanie binarne jest realizowane za pomocą języka Python.
Implementacja wyszukiwania binarnego
W wyszukiwaniu binarnym bierzemy posortowaną listę elementów i zaczynamy szukać elementu na środku listy. Jeśli wartość wyszukiwania odpowiada środkowej wartości na liście, kończymy wyszukiwanie. W przeciwnym razie usuwamy połowę listy elementów, wybierając, czy ma postępować z prawą, czy lewą połową listy w zależności od wartości szukanej pozycji. Jest to możliwe, ponieważ lista jest posortowana i jest znacznie szybsza niż wyszukiwanie liniowe. Tutaj dzielimy podaną listę i podbijamy, wybierając odpowiednią połowę listy. Powtarzamy tę procedurę, aż znajdziemy element lub wyciągniemy wniosek o jego braku na liście.
def bsearch(list, val):
list_size = len(list) - 1
idx0 = 0
idxn = list_size
# Find the middle most value
while idx0 <= idxn:
midval = (idx0 + idxn)// 2
if list[midval] == val:
return midval
# Compare the value the middle most value
if val > list[midval]:
idx0 = midval + 1
else:
idxn = midval - 1
if idx0 > idxn:
return None
# Initialize the sorted list
list = [2,7,19,34,53,72]
# Print the search result
print(bsearch(list,72))
print(bsearch(list,11))
Wykonanie powyższego kodu daje następujący wynik:
5
None