Оптимизации динамического программирования: различия между версиями
Перейти к навигации
Перейти к поиску
Ctrlalt (обсуждение | вклад) Нет описания правки |
Ctrlalt (обсуждение | вклад) Нет описания правки |
||
| (не показаны 2 промежуточные версии этого же участника) | |||
| Строка 1: | Строка 1: | ||
== Divide & Conquer-оптимизация == | |||
Решается задача «Разделить массив на partCount частей, минимизировав нечто». Паттерн — двумерная динамика, позиция в массиве + число. | |||
Базовое решение за O(N^2 * P): | |||
vector<vector<long long>> res(a.size() + 1, vector<long long>(partCount + 1, 1e18)); | |||
res[0][0] = 0; | |||
for (int size = 0; size <= a.size(); size++) { | |||
for (int parts = 1; parts <= partCount; parts++) { | |||
for (int partSize = 1; partSize <= size; partSize++) { | |||
long long partPrice = getPartPrice(size - partSize, size - 1); | |||
res[size][parts] = min(res[size][parts], res[size - partSize][parts - 1] + partPrice); | |||
} | |||
} | |||
} | |||
return res[a.size()][partCount]; | |||
Поменяем порядок индексов и будем перебирать prevSize вместо partSize: | |||
vector<vector<long long>> res(partCount + 1, vector<long long>(a.size() + 1, 1e18)); | |||
res[0][0] = 0; | |||
for (int parts = 1; parts <= partCount; parts++) { | |||
for (int size = 0; size <= a.size(); size++) { | |||
for (int prevSize = 0; prevSize < size; prevSize++) { | |||
long long partPrice = getPartPrice(prevSize, size - 1); | |||
res[parts][size] = min(res[parts][size], res[parts - 1][prevSize] + partPrice); | |||
} | |||
} | |||
} | |||
return res[partCount][a.size()]; | |||
Вынесем расчёт строки для фиксированного parts в функцию: | |||
void calc(vector<vector<long long>> &res, int parts, vector<long long> &p) { | |||
for (int size = 0; size < res[0].size(); size++) { | |||
for (int prevSize = 0; prevSize < size; prevSize++) { | |||
long long partPrice = getPartPrice(prevSize, size - 1); | |||
res[parts][size] = min(res[parts][size], res[parts - 1][prevSize] + partPrice); | |||
} | |||
} | |||
} | |||
vector<vector<long long>> res(partCount + 1, vector<long long>(a.size() + 1, 1e18)); | |||
res[0][0] = 0; | |||
for (int parts = 1; parts <= partCount; parts++) | |||
calc(res, parts, p); | |||
return res[partCount][a.size()]; | |||
Внутри функции вместо цикла по size сделаем рекурсию: | |||
void rec(vector<vector<long long>> &res, int parts, int sizeL, int sizeR, vector<long long> &p) { | |||
if (sizeL > sizeR) | |||
return; | |||
int size = sizeL + (sizeR - sizeL) / 2; | |||
for (int prevSize = 0; prevSize < size; prevSize++) { | |||
long long partPrice = getPartPrice(prevSize, size - 1); | |||
res[parts][size] = min(res[parts][size], res[parts - 1][prevSize] + partPrice); | |||
} | |||
rec(res, parts, sizeL, size - 1, p); | |||
rec(res, parts, size + 1, sizeR, p); | |||
} | |||
vector<vector<long long>> res(partCount + 1, vector<long long>(a.size() + 1, 1e18)); | |||
res[0][0] = 0; | |||
for (int parts = 1; parts <= partCount; parts++) | |||
rec(res, parts, 0, a.size(), p); | |||
return res[partCount][a.size()]; | |||
В рекурсии будем сохранять лучший prevSize: | |||
void rec(vector<vector<long long>> &res, int parts, int sizeL, int sizeR, vector<long long> &p) { | |||
if (sizeL > sizeR) | |||
return; | |||
int size = sizeL + (sizeR - sizeL) / 2; | |||
int bestPrevSize = 0; | |||
for (int prevSize = 0; prevSize < size; prevSize++) { | |||
long long partPrice = getPartPrice(prevSize, size - 1); | |||
if (res[parts][size] > res[parts - 1][prevSize] + partPrice) { | |||
res[parts][size] = res[parts - 1][prevSize] + partPrice; | |||
bestPrevSize = prevSize; | |||
} | |||
} | |||
rec(res, parts, sizeL, size - 1, p); | |||
rec(res, parts, size + 1, sizeR, p); | |||
} | |||
vector<vector<long long>> res(partCount + 1, vector<long long>(a.size() + 1, 1e18)); | |||
res[0][0] = 0; | |||
for (int parts = 1; parts <= partCount; parts++) | |||
rec(res, parts, 0, a.size(), p); | |||
return res[partCount][a.size()]; | |||
Заметим, что в рекурсивных вызовах можно перебирать prevSize только до bestPrevSize или только от bestPrevSize. | |||
Сложность становится равной O(NlogN * P). | |||
void rec(vector<vector<long long>> &res, int parts, int sizeL, int sizeR, int prevSizeL, int prevSizeR, vector<long long> &p) { | |||
if (sizeL > sizeR) | |||
return; | |||
int size = sizeL + (sizeR - sizeL) / 2; | |||
int bestPrevSize = prevSizeL; | |||
for (int prevSize = prevSizeL; prevSize <= prevSizeR && prevSize < size; prevSize++) { | |||
long long partPrice = getPartPrice(p, prevSize, size - 1); | |||
if (res[parts][size] > res[parts - 1][prevSize] + partPrice) { | |||
res[parts][size] = res[parts - 1][prevSize] + partPrice; | |||
bestPrevSize = prevSize; | |||
} | |||
} | |||
rec(res, parts, sizeL, size - 1, prevSizeL, bestPrevSize, p); | |||
rec(res, parts, size + 1, sizeR, bestPrevSize, prevSizeR, p); | |||
} | |||
vector<vector<long long>> res(partCount + 1, vector<long long>(a.size() + 1, 1e18)); | |||
res[0][0] = 0; | |||
for (int parts = 1; parts <= partCount; parts++) | |||
rec(res, parts, 0, a.size(), 0, a.size(), p); | |||
return res[partCount][a.size()]; | |||
* [https://algorithmica.org/ru/dp-optimizations#%D1%80%D0%B0%D0%B7%D0%B4%D0%B5%D0%BB%D1%8F%D0%B9-%D0%B8-%D0%B2%D0%BB%D0%B0%D1%81%D1%82%D0%B2%D1%83%D0%B9 algorithmica.org — Пересчёт динамики по слоям. Разделяй-и-властвуй] | |||
* [http://wiki.algocode.ru/index.php?title=Divide%26Conquer_%D0%BE%D0%BF%D1%82%D0%B8%D0%BC%D0%B8%D0%B7%D0%B0%D1%86%D0%B8%D1%8F algocode.ru — Divide & Conquer-оптимизация] | |||
* [https://cp-algorithms.com/dynamic_programming/divide-and-conquer-dp.html cp-algorithms.com — Divide and Conquer DP] | |||
* [https://elevencompetingnodes.quora.com/DP-Divide-and-Conquer-Optimization Nasser M. — DP Divide and Conquer Optimization] | |||
* [https://robert1003.github.io/2020/02/25/dp-opt-divide-and-conquer.html robert1003.github.io — Divide and Conquer DP] | |||
* [https://usaco.guide/plat/DC-DP usaco.guide — Divide & Conquer DP] | |||
* [https://jeffreyxiao.me/blog/divide-and-conquer-optimization Xiao J. — Divide and Conquer Optimization] | |||
== Ссылки == | == Ссылки == | ||
Теория: | Теория: | ||
* [http://algorithmica.org/ru/dp-optimizations algorithmica.org — Пересчёт динамики по слоям] | * [http://algorithmica.org/ru/dp-optimizations algorithmica.org — Пересчёт динамики по слоям] | ||
* algocode.ru — Оптимизации динамики: [http://wiki.algocode.ru/index.php?title=%D0%9C%D0%BE%D0%BD%D0%BE%D1%82%D0%BE%D0%BD%D0%BD%D0%BE%D1%81%D1%82%D1%8C_%D1%82%D0%BE%D1%87%D0%BA%D0%B8_%D0%BF%D0%B5%D1%80%D0%B5%D0%B3%D0%B8%D0%B1%D0%B0 1 | * algocode.ru — Оптимизации динамики: [http://wiki.algocode.ru/index.php?title=%D0%9C%D0%BE%D0%BD%D0%BE%D1%82%D0%BE%D0%BD%D0%BD%D0%BE%D1%81%D1%82%D1%8C_%D1%82%D0%BE%D1%87%D0%BA%D0%B8_%D0%BF%D0%B5%D1%80%D0%B5%D0%B3%D0%B8%D0%B1%D0%B0 1], [http://wiki.algocode.ru/index.php?title=%D0%9E%D0%BF%D1%82%D0%B8%D0%BC%D0%B8%D0%B7%D0%B0%D1%86%D0%B8%D1%8F_%D0%9A%D0%BD%D1%83%D1%82%D0%B0 2], [http://wiki.algocode.ru/index.php?title=Convex_hull_trick 3], [http://wiki.algocode.ru/index.php?title=%D0%94%D0%B5%D1%80%D0%B5%D0%B2%D0%BE_Li_Chao 4], [http://wiki.algocode.ru/index.php?title=%D0%9B%D1%8F%D0%BC%D0%B1%D0%B4%D0%B0-%D0%BE%D0%BF%D1%82%D0%B8%D0%BC%D0%B8%D0%B7%D0%B0%D1%86%D0%B8%D1%8F 5] | ||
* [https://codeforces.com/blog/entry/8219 codeforces.com — Dynamic Programming Optimizations] | * [https://codeforces.com/blog/entry/8219 codeforces.com — Dynamic Programming Optimizations] | ||
* [https://codeforces.com/blog/entry/47764 codeforces.com — Non-trivial DP Tricks and Techniques] | * [https://codeforces.com/blog/entry/47764 codeforces.com — Non-trivial DP Tricks and Techniques] | ||
* [https://cp-algorithms.com/dynamic_programming/knuth-optimization.html cp-algorithms.com — Knuth's Optimization] | * [https://cp-algorithms.com/dynamic_programming/knuth-optimization.html cp-algorithms.com — Knuth's Optimization] | ||
* [https://usaco.guide/adv/dp-more usaco.guide — Additional DP Optimizations and Techniques] | * [https://usaco.guide/adv/dp-more usaco.guide — Additional DP Optimizations and Techniques] | ||
* [http://jeffe.cs.illinois.edu/teaching/algorithms/notes/D-faster-dynprog.pdf Erickson J. Advanced Dynamic Programming] | * [http://jeffe.cs.illinois.edu/teaching/algorithms/notes/D-faster-dynprog.pdf Erickson J. Advanced Dynamic Programming] | ||
* Xiao J. [https://jeffreyxiao.me/blog/convex-hull-trick Convex Hull Trick], [https://jeffreyxiao.me/blog/knuths-optimization Knuth's | * Xiao J. [https://jeffreyxiao.me/blog/convex-hull-trick Convex Hull Trick], [https://jeffreyxiao.me/blog/knuths-optimization Knuth's Optimization] | ||
* [http://maratona.ic.unicamp.br/MaratonaVerao2017/documents/dp.pdf Ахмедов М. Dynamic programming optimizations] | * [http://maratona.ic.unicamp.br/MaratonaVerao2017/documents/dp.pdf Ахмедов М. Dynamic programming optimizations] | ||
* [http://acm.math.spbu.ru/~sk1/mm/lections/zksh2017-dp/conspect.pdf Копелиович С. Лекция по динамике (ЗКШ 2017)] | * [http://acm.math.spbu.ru/~sk1/mm/lections/zksh2017-dp/conspect.pdf Копелиович С. Лекция по динамике (ЗКШ 2017)] | ||
[[Категория:Динамическое программирование]] | [[Категория:Динамическое программирование]] | ||
Текущая версия от 18:05, 8 июня 2026
Divide & Conquer-оптимизация
Решается задача «Разделить массив на partCount частей, минимизировав нечто». Паттерн — двумерная динамика, позиция в массиве + число.
Базовое решение за O(N^2 * P):
vector<vector<long long>> res(a.size() + 1, vector<long long>(partCount + 1, 1e18));
res[0][0] = 0;
for (int size = 0; size <= a.size(); size++) {
for (int parts = 1; parts <= partCount; parts++) {
for (int partSize = 1; partSize <= size; partSize++) {
long long partPrice = getPartPrice(size - partSize, size - 1);
res[size][parts] = min(res[size][parts], res[size - partSize][parts - 1] + partPrice);
}
}
}
return res[a.size()][partCount];
Поменяем порядок индексов и будем перебирать prevSize вместо partSize:
vector<vector<long long>> res(partCount + 1, vector<long long>(a.size() + 1, 1e18));
res[0][0] = 0;
for (int parts = 1; parts <= partCount; parts++) {
for (int size = 0; size <= a.size(); size++) {
for (int prevSize = 0; prevSize < size; prevSize++) {
long long partPrice = getPartPrice(prevSize, size - 1);
res[parts][size] = min(res[parts][size], res[parts - 1][prevSize] + partPrice);
}
}
}
return res[partCount][a.size()];
Вынесем расчёт строки для фиксированного parts в функцию:
void calc(vector<vector<long long>> &res, int parts, vector<long long> &p) {
for (int size = 0; size < res[0].size(); size++) {
for (int prevSize = 0; prevSize < size; prevSize++) {
long long partPrice = getPartPrice(prevSize, size - 1);
res[parts][size] = min(res[parts][size], res[parts - 1][prevSize] + partPrice);
}
}
}
vector<vector<long long>> res(partCount + 1, vector<long long>(a.size() + 1, 1e18));
res[0][0] = 0;
for (int parts = 1; parts <= partCount; parts++)
calc(res, parts, p);
return res[partCount][a.size()];
Внутри функции вместо цикла по size сделаем рекурсию:
void rec(vector<vector<long long>> &res, int parts, int sizeL, int sizeR, vector<long long> &p) {
if (sizeL > sizeR)
return;
int size = sizeL + (sizeR - sizeL) / 2;
for (int prevSize = 0; prevSize < size; prevSize++) {
long long partPrice = getPartPrice(prevSize, size - 1);
res[parts][size] = min(res[parts][size], res[parts - 1][prevSize] + partPrice);
}
rec(res, parts, sizeL, size - 1, p);
rec(res, parts, size + 1, sizeR, p);
}
vector<vector<long long>> res(partCount + 1, vector<long long>(a.size() + 1, 1e18));
res[0][0] = 0;
for (int parts = 1; parts <= partCount; parts++)
rec(res, parts, 0, a.size(), p);
return res[partCount][a.size()];
В рекурсии будем сохранять лучший prevSize:
void rec(vector<vector<long long>> &res, int parts, int sizeL, int sizeR, vector<long long> &p) {
if (sizeL > sizeR)
return;
int size = sizeL + (sizeR - sizeL) / 2;
int bestPrevSize = 0;
for (int prevSize = 0; prevSize < size; prevSize++) {
long long partPrice = getPartPrice(prevSize, size - 1);
if (res[parts][size] > res[parts - 1][prevSize] + partPrice) {
res[parts][size] = res[parts - 1][prevSize] + partPrice;
bestPrevSize = prevSize;
}
}
rec(res, parts, sizeL, size - 1, p);
rec(res, parts, size + 1, sizeR, p);
}
vector<vector<long long>> res(partCount + 1, vector<long long>(a.size() + 1, 1e18));
res[0][0] = 0;
for (int parts = 1; parts <= partCount; parts++)
rec(res, parts, 0, a.size(), p);
return res[partCount][a.size()];
Заметим, что в рекурсивных вызовах можно перебирать prevSize только до bestPrevSize или только от bestPrevSize.
Сложность становится равной O(NlogN * P).
void rec(vector<vector<long long>> &res, int parts, int sizeL, int sizeR, int prevSizeL, int prevSizeR, vector<long long> &p) {
if (sizeL > sizeR)
return;
int size = sizeL + (sizeR - sizeL) / 2;
int bestPrevSize = prevSizeL;
for (int prevSize = prevSizeL; prevSize <= prevSizeR && prevSize < size; prevSize++) {
long long partPrice = getPartPrice(p, prevSize, size - 1);
if (res[parts][size] > res[parts - 1][prevSize] + partPrice) {
res[parts][size] = res[parts - 1][prevSize] + partPrice;
bestPrevSize = prevSize;
}
}
rec(res, parts, sizeL, size - 1, prevSizeL, bestPrevSize, p);
rec(res, parts, size + 1, sizeR, bestPrevSize, prevSizeR, p);
}
vector<vector<long long>> res(partCount + 1, vector<long long>(a.size() + 1, 1e18));
res[0][0] = 0;
for (int parts = 1; parts <= partCount; parts++)
rec(res, parts, 0, a.size(), 0, a.size(), p);
return res[partCount][a.size()];
- algorithmica.org — Пересчёт динамики по слоям. Разделяй-и-властвуй
- algocode.ru — Divide & Conquer-оптимизация
- cp-algorithms.com — Divide and Conquer DP
- Nasser M. — DP Divide and Conquer Optimization
- robert1003.github.io — Divide and Conquer DP
- usaco.guide — Divide & Conquer DP
- Xiao J. — Divide and Conquer Optimization
Ссылки
Теория:
- algorithmica.org — Пересчёт динамики по слоям
- algocode.ru — Оптимизации динамики: 1, 2, 3, 4, 5
- codeforces.com — Dynamic Programming Optimizations
- codeforces.com — Non-trivial DP Tricks and Techniques
- cp-algorithms.com — Knuth's Optimization
- usaco.guide — Additional DP Optimizations and Techniques
- Erickson J. Advanced Dynamic Programming
- Xiao J. Convex Hull Trick, Knuth's Optimization
- Ахмедов М. Dynamic programming optimizations
- Копелиович С. Лекция по динамике (ЗКШ 2017)