Задача о рюкзаке — различия между версиями

Материал из Викиконспекты
Перейти к: навигация, поиск
м
Строка 1: Строка 1:
'''Задача о рюкзаке''' — дано <tex>k</tex> предметов, <tex>i-ый</tex> предмет имеет массу <tex> w_i > 0</tex> и стоимость <tex> p_i > 0</tex>. Необходимо выбрать из этих предметов такой набор, чтобы суммарная масса не превосходила заданной величины <tex>W</tex> (вместимость рюкзака), а суммарная стоимость была максимальна.
+
'''Задача о рюкзаке''' — дано <tex>k</tex> предметов, <tex>k_i</tex> предмет имеет массу <tex> w_i > 0</tex> и стоимость <tex> p_i > 0</tex>. Необходимо выбрать из этих предметов такой набор, чтобы суммарная масса не превосходила заданной величины <tex>W</tex> (вместимость рюкзака), а суммарная стоимость была максимальна.
  
 
== Формулировка задачи ==
 
== Формулировка задачи ==
Строка 15: Строка 15:
 
* Перебирая все подмножества набора из k предметов. Сложность такого решения <tex>O({2^{k}})</tex>.
 
* Перебирая все подмножества набора из k предметов. Сложность такого решения <tex>O({2^{k}})</tex>.
  
* Методом [[Meet-in-the-middle|Meet-in-the-middle]][http://neerc.ifmo.ru/wiki/index.php?title=Meet-in-the-middle#.D0.97.D0.B0.D0.B4.D0.B0.D1.87.D0.B0_.D0.BE_.D1.80.D1.8E.D0.BA.D0.B7.D0.B0.D0.BA.D0.B5 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(k \times W)</tex>. Рассмотрим этот алгоритм подробнее.
+
* Метод динамического программирования. Сложность - <tex>O(k \times W)</tex>.
  
== Алгоритм <tex>O(k \times W)</tex> ==
+
== Метод динамического программирования ==
  
 
Пусть <tex>A(s, n)</tex> есть максимальная стоимости предметов, которые можно уложить в рюкзак вместимости n, если можно использовать только первые s предметов из заданных k.
 
Пусть <tex>A(s, n)</tex> есть максимальная стоимости предметов, которые можно уложить в рюкзак вместимости n, если можно использовать только первые s предметов из заданных k.
Строка 41: Строка 41:
  
 
== Реализация ==
 
== Реализация ==
Сначала генерируем <tex>А</tex>
+
Сначала генерируем <tex>A</tex>.
  
 
  for i = 0..W
 
  for i = 0..W
Строка 82: Строка 82:
  
 
{| class="wikitable" cellpadding="4" border="1" style="border-collapse: collapse;"
 
{| class="wikitable" cellpadding="4" border="1" style="border-collapse: collapse;"
 +
|-
 +
| || 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 = 0|| 0|| 0|| 0|| 0|| 0|| 0|| 0|| 0|| 0|| 0|| 0|| 0|| 0|| 0
Строка 95: Строка 97:
 
| s = 5|| 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 до 15 в первой строчке обозначают вместимость рюкзака.
+
Числа от 0 до 13 в первой строчке обозначают вместимость рюкзака.
 +
 
 +
В первой строке как только вместимость рюкзака <tex>n \ge 3</tex>, добавляем в рюкзак 1 предмет.
 +
 
 +
Рассмотрим <tex>s = 3</tex>, при каждом <tex>n \ge 5 (</tex>так как <tex>w_3 = 5)</tex> сравниваем <tex>A[s-1][n] и A[s-1][n-w_3]+p_3</tex> и записываем в <tex>A[s][n]</tex> стоимость либо рюкзака без третьего предмета, но с таким же весом, либо с третьим предметом, тогда стоимость равна стоимость третьего предмета плюс стоимость рюкзака с вместимостью на <tex>w_3</tex> меньше.   
 +
 
 +
Максимальная стоимость рюкзака находится в <tex>A(5, 13)</tex>.
 +
 
 +
'''Теперь восстановим набор предметов, из которых состоит максимально дорогой рюкзак.'''
 +
 
 +
 
 +
 
 +
 
 +
 
 +
 
 +
 
 +
 
 +
 
 +
 
 +
 
 +
 
 +
 
 +
 
  
В первой строке как только вместимость рюкзака <tex>n /ge 3</tex>, добавляем в рюкзак 1 предмет.
 
  
Во второй строке, когда можно использовать первые <tex>2</tex> предмета.
 
  
  
  
Максимальная стоимость рюкзака находится в <tex>A(5, 15)</tex>.
 
  
'''Теперь восстановим набор предметов, из которых состоит максимально дорогой рюкзак.'''
 
  
Сравниваем <tex>A(5, 15) = 14</tex> и <tex>A(4, 15) = 12</tex>. Не равны. Следовательно, <tex>5</tex> предмет входит в искомый набор, переходим к <tex>4</tex> предмету с весом рюкзака <tex>W - w_5</tex>. То есть <tex>15 - 5 = 10</tex>
 
  
Сравниваем <tex>A(4, 10) = 8</tex> и <tex>A(3, 10) = 8</tex>. Равны. Следовательно, <tex>4</tex> предмет не входит в набор, переходим к <tex>3</tex> предмету с тем же весом рюкзака.
 
  
Сравниваем <tex>A(3, 10) = 8</tex> и <tex>A(2, 10) = 8</tex>. Равны. Следовательно, <tex>4</tex> предмет не входит в набор, переходим к <tex>2</tex> предмету с тем же весом рюкзака.
 
  
Сравниваем <tex>A(2, 10) = 2</tex> и <tex>A(1, 10) = 5</tex>. Не равны. Следовательно, <tex>2</tex> предмет входит в набор, уменьшаем вес рюкзака на <tex>w_2</tex>, переходим к <tex>1</tex> предмету.
 
  
Сравниваем <tex>A(1, 6) = 5</tex> и <tex>A(0, 6) = 0</tex>. Не равны. Следовательно, <tex>1</tex> предмет входит в набор.
 
  
 
Таким образом, набор состоит из <tex>1, 2, 5</tex> предметов.
 
Таким образом, набор состоит из <tex>1, 2, 5</tex> предметов.

Версия 14:12, 27 декабря 2012

Задача о рюкзаке — дано [math]k[/math] предметов, [math]k_i[/math] предмет имеет массу [math] w_i \gt 0[/math] и стоимость [math] p_i \gt 0[/math]. Необходимо выбрать из этих предметов такой набор, чтобы суммарная масса не превосходила заданной величины [math]W[/math] (вместимость рюкзака), а суммарная стоимость была максимальна.

Формулировка задачи

Дано [math]k[/math] предметов, [math]W[/math] - вместимость рюкзака, [math]w=\{w_{1},w_{2},...,w_{k}\}[/math] — соответствующий ему набор положительных целых весов, [math]p=\{p_{1},p_{2},...,p_{k}\}[/math] — соответствующий ему набор положительных целых стоимостей. Нужно определить набор бинарных величин [math]B=\{b_{1},b_{2},...,b_{k}\}[/math], где [math]b_{i} = 1 [/math], если предмет включен в набор, [math] b_{i} = 0 [/math], если предмет не включен, такой что:

  1. [math]b_{1} w_{1}+ ... + b_{k} w_{k} \le W[/math]
  1. [math]b_{1} p_{1}+ ... + b_{k} k_{k} [/math] максимальна.

Варианты решения

Задачу о рюкзаке можно решить несколькими способами:

  • Перебирая все подмножества набора из k предметов. Сложность такого решения [math]O({2^{k}})[/math].
  • Методом Meet-in-the-middle. Сложность решения [math] O({2^{N/2}}\times{N}) [/math]
  • Метод динамического программирования. Сложность - [math]O(k \times W)[/math].

Метод динамического программирования

Пусть [math]A(s, n)[/math] есть максимальная стоимости предметов, которые можно уложить в рюкзак вместимости n, если можно использовать только первые s предметов из заданных k.

[math]A(s, 0) = 0[/math]

[math]A(0, n) = 0[/math]

Найдем [math]A(s, n)[/math]. Возможны 2 варианта:

  1. Если предмет [math]s[/math] не попал в рюкзак. Тогда [math]A(s, n) = A(s-1, n)[/math]
  1. Если [math]s[/math] попал в рюкзак. Тогда [math]A(s, n) = A(s-1, n-w_{s}) + p_{s}[/math]

Таким образом: [math]A(s,n) = max(A(s-1,n), A(s-1,n-w_{s}) + p_{s})[/math]

Теперь найдем набор предметов, входящих в рюкзак.

Рассмотрим входит ли [math]k[/math] - последний предмет в рюкзак. Если [math]A(k, W)[/math] равно [math]A(k-1, W)[/math], значит последний предмет не входит в набор, иначе входит. Так рекусривно идем до первого предмета. Получаем искомый набор.

Реализация

Сначала генерируем [math]A[/math].

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]             //иначе, не кладем

Затем найдем набор [math]ans[/math] предметов, входящих в рюкзак, рекурсивной функцией:

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);

Сложность алгоритма [math]O(kW)[/math]

Пример

[math]W = 13, k = 5[/math]

[math]w_{1} = 3, p_{1} = 1 [/math]

[math]w_{2} = 4, p_{2} = 6 [/math]

[math]w_{3} = 5, p_{3} = 4 [/math]

[math]w_{4} = 8, p_{4} = 7 [/math]

[math]w_{5} = 9, p_{5} = 6 [/math]

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 в первой строчке обозначают вместимость рюкзака.

В первой строке как только вместимость рюкзака [math]n \ge 3[/math], добавляем в рюкзак 1 предмет.

Рассмотрим [math]s = 3[/math], при каждом [math]n \ge 5 ([/math]так как [math]w_3 = 5)[/math] сравниваем [math]A[s-1][n] и A[s-1][n-w_3]+p_3[/math] и записываем в [math]A[s][n][/math] стоимость либо рюкзака без третьего предмета, но с таким же весом, либо с третьим предметом, тогда стоимость равна стоимость третьего предмета плюс стоимость рюкзака с вместимостью на [math]w_3[/math] меньше.

Максимальная стоимость рюкзака находится в [math]A(5, 13)[/math].

Теперь восстановим набор предметов, из которых состоит максимально дорогой рюкзак.














Таким образом, набор состоит из [math]1, 2, 5[/math] предметов.

Стоимость рюкзака [math]= 5 + 3 + 6 = 14[/math]

Вес рюкзака [math]= 6 + 4 + 5 = 15[/math]

Литература