Задача о рюкзаке — различия между версиями
| Строка 1: | Строка 1: | ||
| − | '''Задача о рюкзаке''' — дано <tex> | + | '''Задача о рюкзаке''' — дано <tex>N</tex> предметов, <tex>n_i</tex> предмет имеет массу <tex> w_i > 0</tex> и стоимость <tex> p_i > 0</tex>. Необходимо выбрать из этих предметов такой набор, чтобы суммарная масса не превосходила заданной величины <tex>W</tex> (вместимость рюкзака), а суммарная стоимость была максимальна. |
== Формулировка задачи == | == Формулировка задачи == | ||
| − | Дано <tex> | + | Дано <tex>N</tex> предметов, <tex>W</tex> - вместимость рюкзака, <tex>w=\{w_{1},w_{2},...,w_{N}\}</tex> — соответствующий ему набор положительных целых весов, <tex>p=\{p_{1},p_{2},...,p_{N}\}</tex> — соответствующий ему набор положительных целых стоимостей. Нужно найти набор бинарных величин <tex>B=\{b_{1},b_{2},...,b_{N}\}</tex>, где <tex>b_{i} = 1 </tex>, если предмет <tex>n_i</tex> включен в набор, <tex> b_{i} = 0 </tex>, если предмет <tex>n_i</tex> не включен, и такой что: |
| − | # <tex>b_{1} w_{1}+ ... + b_{ | + | # <tex>b_{1} w_{1}+ ... + b_{N} w_{N} \le W</tex> |
| − | # <tex>b_{1} p_{1}+ ... + b_{ | + | # <tex>b_{1} p_{1}+ ... + b_{N} k_{N} </tex> максимальна. |
== Варианты решения == | == Варианты решения == | ||
| Строка 13: | Строка 13: | ||
'''Задачу о рюкзаке можно решить несколькими способами:''' | '''Задачу о рюкзаке можно решить несколькими способами:''' | ||
| − | * Перебирая все подмножества набора из | + | * Перебирая все подмножества набора из N предметов. Сложность такого решения <tex>O({2^{N}})</tex>. |
* Методом [[Meet-in-the-middle|Meet-in-the-middle]]. Сложность решения <tex> O({2^{N/2}}\times{N}) </tex> | * Методом [[Meet-in-the-middle|Meet-in-the-middle]]. Сложность решения <tex> O({2^{N/2}}\times{N}) </tex> | ||
| − | * Метод динамического программирования. Сложность - <tex>O( | + | * Метод динамического программирования. Сложность - <tex>O(N \times W)</tex>. |
== Метод динамического программирования == | == Метод динамического программирования == | ||
| − | + | Пусть <tex>A(k, s)</tex> есть максимальная стоимости предметов, которые можно уложить в рюкзак вместимости <tex>s</tex>, если можно использовать только первые <tex>k</tex> предметов, то есть <tex>\{n_1,n_2,...,n_k\}</tex>, назовем этот набор допустимых предметов. | |
| + | |||
| + | <tex>A(k, 0) = 0</tex> | ||
| + | |||
| + | <tex>A(0, s) = 0</tex> | ||
| + | |||
| + | Найдем <tex>A(k, s)</tex>. Возможны 2 варианта: | ||
| + | |||
| + | # Если предмет <tex>k</tex> не попал в рюкзак. Тогда <tex>A(k, s)</tex> равно максимальной стоимости, с такой же вместимостью и набором допустимых предметов <tex>\{n_1,n_2,...,n_(k-1)\}</tex>, то есть <tex>A(k,s)</tex> = A(k-1, s)</tex> | ||
| + | |||
| + | # Если <tex>k</tex> попал в рюкзак. Тогда <tex>A(k, s)</tex> равно максимальной стоимоти, где вес s уменьшаем на вес k рюкзака и набор допустимых предметов <tex>\{n_1,n_2,...,n_(k-1)\}</tex> плюс стоимость <tex>k рюкзака, то есть <tex>A(k-1, s-w_k) + p_k</tex> | ||
| + | |||
| + | Если короче: | ||
| + | |||
| + | # <tex>A(k,s) = A(k-1, s)</tex> | ||
| + | |||
| + | # <tex>A(k,s) = A(k-1, s-w_k) + p_k</tex> | ||
| + | |||
| + | Выберем из этих двух значений максимальное: | ||
| + | |||
| + | <tex>A(k,s) = max(A(k-1,s), A(k-1,s-w_{k}) + p_{k})</tex> | ||
| + | |||
| + | '''Восстановим набор предметов, входящих в рюкзак.''' | ||
| + | |||
| + | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
| − | |||
== Реализация == | == Реализация == | ||
Версия 11:44, 29 декабря 2012
Задача о рюкзаке — дано предметов, предмет имеет массу и стоимость . Необходимо выбрать из этих предметов такой набор, чтобы суммарная масса не превосходила заданной величины (вместимость рюкзака), а суммарная стоимость была максимальна.
Содержание
Формулировка задачи
Дано предметов, - вместимость рюкзака, — соответствующий ему набор положительных целых весов, — соответствующий ему набор положительных целых стоимостей. Нужно найти набор бинарных величин , где , если предмет включен в набор, , если предмет не включен, и такой что:
- максимальна.
Варианты решения
Задачу о рюкзаке можно решить несколькими способами:
- Перебирая все подмножества набора из N предметов. Сложность такого решения .
- Методом Meet-in-the-middle. Сложность решения
- Метод динамического программирования. Сложность - .
Метод динамического программирования
Пусть есть максимальная стоимости предметов, которые можно уложить в рюкзак вместимости , если можно использовать только первые предметов, то есть , назовем этот набор допустимых предметов.
Найдем . Возможны 2 варианта:
- Если предмет не попал в рюкзак. Тогда равно максимальной стоимости, с такой же вместимостью и набором допустимых предметов , то есть = A(k-1, s)</tex>
- Если попал в рюкзак. Тогда равно максимальной стоимоти, где вес s уменьшаем на вес k рюкзака и набор допустимых предметов плюс стоимость
Если короче:
Выберем из этих двух значений максимальное:
Восстановим набор предметов, входящих в рюкзак.
Реализация
Сначала генерируем .
for i = 0..W
A[0][i] = 0
for i = 0..k
A[i][0] = 0 //Первые элементы приравниваем 0
for s = 1..k
for n = 0..W //Перебираем для каждого s, все n
if n >= w[s] //Если текущий предмет вмещается в рюкзак
A[s][n] = max(A[s-1][n], A[s-1][n-w[s]]+p[s]) //выбираем класть его или нет
else
A[s][n] = A[s-1][n] //иначе, не кладем
Затем найдем набор предметов, входящих в рюкзак, рекурсивной функцией:
findAns(s, n)
if A[s][n] == 0
return
if A[s-1][n] == A[s][n]
findAns(s-1, n)
else
findAns(s-1, n - w[s]);
ans.push(s);
Сложность алгоритма
Пример
| 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | |
| s = 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 | 0 |
| s = 1 | 0 | 0 | 0 | 1 | 1 | 1 | 1 | 1 | 1 | 1 | 1 | 1 | 1 | 1 |
| s = 2 | 0 | 0 | 0 | 1 | 6 | 6 | 6 | 7 | 7 | 7 | 7 | 7 | 7 | 7 |
| s = 3 | 0 | 0 | 0 | 1 | 6 | 6 | 6 | 7 | 7 | 10 | 10 | 10 | 11 | 11 |
| s = 4 | 0 | 0 | 0 | 1 | 6 | 6 | 6 | 7 | 7 | 10 | 10 | 10 | 13 | 13 |
| s = 5 | 0 | 0 | 0 | 1 | 6 | 6 | 6 | 7 | 7 | 10 | 10 | 10 | 13 | 13 |
Числа от 0 до 13 в первой строчке обозначают вместимость рюкзака.
В первой строке как только вместимость рюкзака , добавляем в рюкзак 1 предмет.
Рассмотрим , при каждом так как сравниваем и записываем в стоимость либо рюкзака без третьего предмета, но с таким же весом, либо с третьим предметом, тогда стоимость равна стоимость третьего предмета плюс стоимость рюкзака с вместимостью на меньше.
Максимальная стоимость рюкзака находится в .
Восстановление набора предметов, из которых состоит максимально дорогой рюкзак.
Будем начиная с ячейки, соответствующей ответу, то есть вместимость , и можно использовать для составления набора все предметы, определять, входит последний предмет в набор или нет. Для этого сравниваем значение в рассматриваемой ячейке с:
- Со значением в ячейке с такой же вместимостью, и можно использовать все предметы имеющие
Таким образом, набор состоит из предметов.
Стоимость рюкзака
Вес рюкзака