Хотя я понимаю, что вы не хотите, чтобы пробный прогон просто подсчитывал звонки, я все же хотел бы сначала сделать это, просто чтобы посмотреть, что происходит.Итак, вот так:
def min_cost(s, d):
global count
count += 1
if s==d or s == d-1:
return cost[s][d]
mc = cost[s][d]
for i in range(s+1, d):
tmp = min_cost(s, i) + min_cost(i, d)
if tmp < mc:
mc = tmp
return mc
for n in range (2, 8):
cost = [[0 for i in range (n)] for j in range (n)]
count = 0
min_cost(0, n-1)
print (str (n) + ': ' + str (count))
Вывод:
2: 1
3: 3
4: 9
5: 27
6: 81
7: 243
Итак, мы видим, что количество вызовов для ds = k равно 3 степени (k-1).Знание того, что мы должны доказать, иногда очень упрощает нахождение доказательства.
Теперь к самому доказательству.Это будет доказательство по индукции .Во-первых, обратите внимание, что количество вызовов min_cost(s, d)
зависит только от значения d-s
, а не от отдельных значений s
и d
.
Основа такова, что для d-s=1
, мы получаем один звонок.Для d-s>1
мы делаем один звонок, и из него следующие звонки:
min_cost(s, s+1) and min_cost(s+1, d)
min_cost(s, s+2) and min_cost(s+2, d)
...
min_cost(s, d-1) and min_cost(d-1, d)
Итак, для d-s=k
количество звонков f(k)
равно:
f(k) = 1 +
f(1) + f(k-1) +
f(2) + f(k-2) +
... +
f(k-1) + f(1)
= 2 * (f(1) + f(2) + ... + f(k-1))
Если по предположению индукции мы уже доказали, что f (v) = 3 v для всех v 1 + 3 2 + ... + 3 k-1 ), что составляет тривиально 3 k , завершая наше доказательство.
Наконец, обратите внимание, что, хотя представленный алгоритм является экспоненциальным, основная задача может быть решена за полиномиальное время, наиболее просто за O ((ds)^ 2) запоминая звонки, для которых мы уже сделали всю работу.