Problema dello zaino

Nessuna nota a piè di pagina
Questa voce o sezione sull'argomento matematica è priva o carente di note e riferimenti bibliografici puntuali.
In questo caso, la soluzione è di mettere nello zaino tre libri gialli e tre grigi

Il problema dello zaino, o in inglese Knapsack problem, è un problema di ottimizzazione combinatoria posto nel modo seguente.

Sia dato uno zaino che possa sopportare un determinato peso e siano dati N {\displaystyle N} oggetti, ognuno dei quali caratterizzato da un peso e un valore. Il problema si propone di scegliere quali di questi oggetti mettere nello zaino per ottenere il maggiore valore senza eccedere il peso sostenibile dallo zaino stesso.

Introduzione

Il problema espresso in maniera più formale diventa:

  • ognuno degli N {\displaystyle N} oggetti possiede un peso w i {\displaystyle w_{i}} e un valore c i {\displaystyle c_{i}} ;
  • si indica con W {\displaystyle W} il peso massimo sopportabile dallo zaino;
  • la possibilità che un oggetto venga inserito o meno nello zaino è espressa dalle variabili intere x i {\displaystyle x_{i}} .

La funzione obiettivo da massimizzare è:

Z = i = 1 N c i x i . {\displaystyle Z=\sum _{i=1}^{N}c_{i}\cdot x_{i}.}

I vincoli:

i = 1 N w i x i W . {\displaystyle \sum _{i=1}^{N}w_{i}\cdot x_{i}\leq W.}

In base al tipo di variabili x i {\displaystyle x_{i}} si ha poi la distinzione in:

  • problema dello zaino 0-1
x i = { 0 , 1 } i = 1 , , N , {\displaystyle x_{i}=\left\{0,1\right\}\quad \forall i=1,\ldots ,N,}
ogni oggetto può esserci o non esserci senza ripetizioni (abbiamo un solo esemplare di ciascun oggetto);
  • problema dello zaino con limiti
x i b i i = 1 , , N , {\displaystyle x_{i}\leq b_{i}\quad \forall i=1,\ldots ,N,}
ogni oggetto può apparire nello zaino al massimo b i {\displaystyle b_{i}} volte (abbiamo b 1 {\displaystyle b_{1}} esemplari dell'oggetto 1, b 2 {\displaystyle b_{2}} esemplari dell'oggetto 2 e così via);
  • problema dello zaino senza limiti
x i N i = 1 , , N , {\displaystyle x_{i}\in \mathbb {N} \quad \forall i=1,\ldots ,N,}
ogni oggetto può apparire nello zaino un numero arbitrario di volte.

Il problema dello zaino è risolto spesso usando la programmazione dinamica, anche se è noto che questo metodo ha un tempo di risoluzione non lineare per questo genere di problema. Il problema generale dello zaino è un problema NP-difficile e questo ha indirizzato la ricerca verso il problema Subset-sum come base per il sistema di crittografia a chiave pubblica, come Merkle-Hellman. Questi tentativi usavano tipicamente alcuni gruppi oltre agli interi. Merkle-Hellman e altri algoritmi simili vennero presto abbandonati poiché i sottoproblemi di somma che producevano erano risolvibili da algoritmi lineari.

La versione decisionale di questo problema è NP-completa e infatti è uno dei 21 problemi NP-completi di Karp.

Il problema dello zaino, nella versione di ottimizzazione, è di fondamentale importanza in quanto può essere risolto in maniera soddisfacente in molti casi di comune applicazione; infatti per questo problema sono disponibili buone euristiche e buoni rilassamenti. Un algoritmo di enumerazione implicita, ad esempio Branch and bound, normalmente non impiega molto tempo per risolverlo.

Soluzione problema dello zaino senza limiti

Viene descritta di seguito la soluzione per il problema dello zaino senza limiti.

Si indichino con c 1 , , c N {\displaystyle c_{1},\ldots ,c_{N}} i guadagni offerti dagli oggetti, e con w 1 , , w N {\displaystyle w_{1},\ldots ,w_{N}} i pesi di ogni oggetto. Si desidera massimizzare il guadagno complessivo mantenendo il peso complessivo minore o uguale al peso massimo consentito W {\displaystyle W} (vincolo). Si indichi con A ( k ) {\displaystyle A(k)} il valore massimo di guadagno che si può ottenere rispettando il vincolo che il peso complessivo sia minore o uguale a k {\displaystyle k} . Ovviamente k W {\displaystyle k\leq W} e A ( W ) {\displaystyle A(W)} è la soluzione del problema.

Si definiscono gli A ( k ) {\displaystyle A(k)} ricorsivamente come di seguito:

  • A ( 0 ) = 0 ; {\displaystyle A(0)=0;}
  • A ( k ) = max { c j + A ( k w j ) | w j k } . {\displaystyle A(k)=\max \lbrace c_{j}+A(k-w_{j})|w_{j}\leq k\rbrace .}

avendo considerato zero il massimo dell'insieme vuoto. Se si tabulano i risultati a partire da A ( 0 ) {\displaystyle A(0)} fino a A ( W ) {\displaystyle A(W)} si ottiene la soluzione. Dato che il calcolo di ogni A ( k ) {\displaystyle A(k)} implica l'esame di N {\displaystyle N} oggetti, tutti calcolati in precedenza, e visto che ci sono W {\displaystyle W} valori di A ( k ) {\displaystyle A(k)} da calcolare, il tempo impiegato per trovare la soluzione è O ( N W ) {\displaystyle O(NW)} .

Ciò non contraddice il fatto che il problema dello zaino sia NP-completo, dato che W {\displaystyle W} , al contrario di N {\displaystyle N} , non è polinomiale rispetto alla lunghezza dell'input del problema. Questa lunghezza è proporzionale al numero di bit in W {\displaystyle W} , e non a W {\displaystyle W} stesso.

Soluzione problema dello zaino 0-1

Si indichino con w i {\displaystyle w_{i}} il peso dell'i-esimo oggetto e con c i {\displaystyle c_{i}} il suo valore. Si vuole massimizzare il valore totale rispettando il vincolo che il peso totale sia minore o uguale al peso massimo consentito W {\displaystyle W} . Definiamo A ( i , j ) {\displaystyle A(i,j)} come il massimo valore che può essere trasportato con uno zaino di capacità j W {\displaystyle j\leq W} avendo a disposizione solo i primi i {\displaystyle i} oggetti.

Si può definire A ( i , j ) {\displaystyle A(i,j)} ricorsivamente come segue:

  • A ( 0 , j ) = 0 , {\displaystyle A(0,j)=0,}
  • A ( i , 0 ) = 0 , {\displaystyle A(i,0)=0,}
  • A ( i , j ) = A ( i 1 , j ) {\displaystyle A(i,j)=A(i-1,j)} se w i > j , {\displaystyle w_{i}>j,}
  • A ( i , j ) = max { A ( i 1 , j ) , A ( i 1 , j w i ) + c i } {\displaystyle A(i,j)=\max \lbrace A(i-1,j),A(i-1,j-w_{i})+c_{i}\rbrace } se w i j . {\displaystyle w_{i}\leq j.}

Si può trovare la soluzione calcolando A ( n , W ) {\displaystyle A(n,W)} . Per farlo in modo efficiente si può usare una tabella che memorizzi i calcoli fatti precedentemente. Questa soluzione impiegherà quindi un tempo proporzionale a O ( n W ) {\displaystyle O(nW)} e uno spazio anch'esso proporzionale a O ( n W ) {\displaystyle O(nW)} , anche se con alcune piccole modifiche si può ridurre lo spazio utilizzato a O ( W ) {\displaystyle O(W)} .

Algoritmo Greedy

Martello e Toth (1990) hanno utilizzato un'euristica greedy per risolvere il problema dello zaino. La loro versione ordina gli oggetti in base al loro costo unitario, vale a dire c i w i {\displaystyle {\frac {c_{i}}{w_{i}}}} e li esamina in ordine decrescente. L'oggetto corrente viene inserito se e solo se il suo peso non supera la capacità residua corrente.

Questi algoritmi sono euristici, quindi non garantiscono di trovare la soluzione ottima, ma sono in grado di fornire una "buona" soluzione in tempo ragionevole; spesso questo tipo di algoritmi viene utilizzato in approcci di enumerazione implicita come gli algoritmi Branch and bound.

Rilassamento continuo

Si dimostra che il rilassamento continuo del problema dello zaino è equivalente all'euristica CUD quando si permettono valori in [ 0 , 1 ] {\displaystyle [0,1]} delle variabili x i {\displaystyle x_{i}} , in particolare una sola variabile avrà valore non binario. In questo modo euristica e rilassamento possono essere risolti simultaneamente in maniera efficiente.

Bibliografia

  • Michael R. Garey, David S. Johnson, Computers and Intractability: A Guide to the Theory of NP-Completeness, W.H. Freeman, 1979, ISBN 0-7167-1045-5.
  • Silvano Martello, Paolo Toth, Knapsack Problems: Algorithms and Computer Implementations, Wiley, 1990, ISBN 0-471-92420-2.

Collegamenti esterni

  Portale Informatica
  Portale Matematica