Мы можем реализовать алгоритм динамического программирования, по крайней мере, в двух разных подходах. Один из них - подход сверху вниз с использованием memoization, другой - итеративный подход снизу вверх.
Для начинающих для динамического программирования я всегда рекомендую сначала использовать подход сверху вниз, так как это поможет им понять отношения повторения в динамическом программировании.
Так что для того, чтобы решить эту монету изменяющейся проблемы, вы уже поняли, что говорят отношения рецидивов:
table[i][j] = table[i-S[j]][j] + table[i][j-1]
Таких отношения повторения это хорошо, но это не то, что хорошо определены, так как он не имеют любые граничные условия. Поэтому нам необходимо определить граничные условия, чтобы гарантировать, что рекуррентное соотношение может успешно завершиться без перехода в бесконечный цикл.
Так что будет, когда мы попытаемся спуститься по рекурсивному дереву?
Если нам нужно вычислить table[i][j]
, что означает число подходов для изменения i
с использованием монет от типа 0
к j
, есть несколько случаев угла, мы должны обращаться:
1) Что делать, если j == 0
?
Если j == 0
мы постараемся решить подзапрос table(i,j-1)
, что является недопустимой подзадачей.Таким образом, один граничное условие:
if(j==0) {
if(i==0) table[i][j] = 1;
else table[i][j] = 0;
}
2) Что делать, если i - S[j] < 0
?
Нам также необходимо обработать этот граничный случай, и мы знаем, что в таком состоянии мы не должны пытаться решить эту подзадачу или инициализировать table(i-S[j],j) = 0
для всех этих случаев.
Итак во всем, если мы собираемся реализовать это динамическое программирование от запоминания подхода сверху вниз, мы можем сделать что-то вроде этого:
int f(int i, int j) {
if(calc[i][j]) return table[i][j];
calc[i][j] = true;
if(j==0) {
if(i==0) return table[i][j]=1;
else return table[i][j]=0;
}
if(i>=S[j])
return table[i][j]=table[i-S[j][j]+table[i][j-1];
else
return table[i][j]=table[i][j-1];
}
На практике, это также возможно, что мы используем значение из table
массивов, чтобы помочь отслеживать, была ли эта подзадача ранее вычислена (например, мы можем инициализировать значение -1, эта подзадача не была рассчитана).
Надеюсь, что ответ ясен. :)
спасибо за разъяснение. Два момента: откуда именно «вычет»? Как логическое условие, где он умирает, он когда-либо установлен как ложный? Во-вторых, как 2-мерный массив, как я могу извлечь возможные последовательности? – Jason
@Jason: 'calc' - это логическая матрица типа, которая обозначает, была ли эта« подзадача »рассчитана раньше, если это так, то нам не нужно ее пересчитывать, и мы можем просто использовать предыдущий вычисленный результат. Это ядро динамического программирования – derekhh
@Jason: Извлечение возможных последовательностей - это совершенно другая проблема, хотя число возможностей может быть довольно огромным, но вы можете использовать вектор v [i] [j], чтобы отслеживать это, следуя аналогичные отношения. –
derekhh