動態規劃算法深度解析:0-1背包問題(含完整流程)

簡介:

0-1背包問題是經典的組合優化問題:給定一組物品(每個物品有重量和價值),在背包容量限制下選擇物品裝入背包,要求總價值最大化每個物品不可重復選取

動態規劃核心思想

通過構建二維狀態表dp[i][j],記錄前i個物品在容量j時的最大價值,通過狀態轉移方程逐步推導最優解,避免重復計算子問題。


問題建模與參數定義
static final Integer N = 4;       // 物品數量
static final Integer W = 5;       // 背包容量
Integer[] w = {0,1,2,3,4};        // 物品重量數組(索引0占位)
Integer[] v = {0,2,4,5,6};        // 物品價值數組
private Integer[][] table = new Integer[N+1][W+1]; // DP狀態表

代碼執行全流程解析

1. 初始化階段 init()
for(int i=0;i<=N;i++) {for(int j=0;j<=W;j++) {table[i][j]=0;}
}

🔍 執行過程

  1. 創建(N+1)行×(W+1)列的二維數組
  2. 初始化邊界條件:
    • table[0][j] = 0(無物品可裝)
    • table[i][0] = 0(無容量可用)
┌───────────────┐
│  Start Init  │
└───────┬───────┘│
┌───────▼───────┐
│ i=0 to N     │
├───────┬───────┤
│ j=0 to W     │
├───────▼───────┤
│ table[i][j]=0 │
└───────┬───────┘│
┌───────▼───────┐
│ End Init      │
└───────────────┘

2. 動態規劃核心 dynamics()
for(int i=1;i<=N;i++) {          // 物品維度for(int j=1;j<=W;j++) {      // 容量維度// 不選當前物品table[i][j] = table[i-1][j]; // 選當前物品(需容量足夠)if(j >= w[i]) {table[i][j] = max(table[i][j], table[i-1][j-w[i]] + v[i]);}}
}

📊 狀態轉移矩陣演變

迭代過程示例(i=2時):容量 j | 0 1 2 3 4 5
i=0     | 0 0 0 0 0 0
i=1     | 0 2 2 2 2 2 
i=2     | 0 2 max(2,2+4)=6 ...

完整流程圖與時序圖

系統級流程圖

在這里插入圖片描述

時序圖


復雜度深度分析

時間復雜度

  • 雙重循環:O(N*W) = 4×5 = 20次核心計算
  • 計算過程:
Σ(i=1→4) Σ(j=1→5) [1次比較 + 1次查詢] = 4×5×2 = 40次操作

空間復雜度

  • 二維數組存儲:O(N*W) = 5×6 = 30個存儲單元
  • 空間消耗分解:
基礎類型Integer × 30 = 30×4 bytes = 120 bytes

完整代碼

public class Knapsack {/** 假設有背包中可以最多可以裝4個產品;背包承受的最大容量為5,求該背包最大的價值為多少* N:為物品數量* W:為背包容量* w[]:表示每一個產品容量* v[]:表示每一個產品的價值** */static final Integer N =4;static final Integer W= 5;Integer[] w =new Integer[]{0,1,2,3,4};Integer[] v= new  Integer[]{0,2,4,5,6};private Integer[][] table = new  Integer[N+1][W+1];void  init(){for(int i=0;i<=N;i++){for(int j=0;j<=W;j++){table[i][j]=0;}}}void print(){for(int i=0;i<=N;i++){for(int j=0;j<=W;j++){System.out.print(table[i][j]+"   ");}System.out.println();}}void dynamics(){for(int i=1;i<=N;i++){for(int j=1;j<=W;j++){table[i][j]=table[i-1][j]; // 不選第i個物品if(j>=w[i]){// 選第i個物品table[i][j]=max(table[i][j],table[i-1][j-w[i]]+v[i]);}}}}// 判斷大小的方法Integer max(Integer value1,Integer value2){return value1>value2?value1:value2;}public static void main(String[] args) {Knapsack k=new Knapsack();k.init();k.dynamics();k.print();}
}

結果截圖:

擴展解法對比

1. 回溯法(決策樹實現)
int backtrack(int i, int currentW, int currentV) {if(i > N) return currentV;if(currentW + w[i] > W) {return backtrack(i+1, currentW, currentV);}return max(backtrack(i+1, currentW, currentV),backtrack(i+1, currentW + w[i], currentV + v[i]));
}

?? 問題規模達20時計算量超百萬次

2. 空間優化DP(滾動數組)
int[] dp = new int[W+1];
for(int i=1; i<=N; i++){for(int j=W; j>=w[i]; j--){ // 逆序更新dp[j] = Math.max(dp[j], dp[j-w[i]] + v[i]);}
}

🔧 優勢:空間復雜度降為O(W) = 6 units

3. 分支限界法(優先隊列實現)
from queue import PriorityQueueclass Node:def __init__(self, level, weight, value, bound):self.level = levelself.weight = weightself.value = valueself.bound = bounddef bound(node):# 計算剩余物品的最大可能價值...

算法選擇策略

方法適用場景時間復雜度空間復雜度
標準動態規劃中小規模精確計算O(N*W)O(N*W)
空間優化DP大規模數據處理O(N*W)O(W)
回溯法物品數<20O(2^N)O(N)
分支限界法需要快速近似解O(2^N)O(2^N)

完整代碼執行結果
0   0   0   0   0   0   
0   2   2   2   2   2   
0   2   4   6   6   6   
0   2   4   6   7   9   
0   2   4   6   7   9   

最終最大價值為 9,通過物品選擇(2+3號物品:重量2+3=5,價值4+5=9)實現


本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/bicheng/76146.shtml
繁體地址,請注明出處:http://hk.pswp.cn/bicheng/76146.shtml
英文地址,請注明出處:http://en.pswp.cn/bicheng/76146.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

ABAP,PDF,ADS,FORM,PRINT

ABAP怎么直接打印PDF文件? https://faskomyabap.blogspot.com/2017/10/how-to-print-pdf-file-content-from-abap.html 里面的程序可以直接將本地文件打印出來,讀一下過程,這個程序是把本地PDF文件使用upload函數到ABAP中,先是二進制,然后轉成XSTRING,然后使用 連招 ADS…

C++Cherno 學習筆記day17 [66]-[70] 類型雙關、聯合體、虛析構函數、類型轉換、條件與操作斷點

b站Cherno的課[66]-[70] 一、C的類型雙關二、C的union&#xff08;聯合體、共用體&#xff09;三、C的虛析構函數四、C的類型轉換五、條件與操作斷點——VisualStudio小技巧 一、C的類型雙關 作用&#xff1a;在C中繞過類型系統 C是強類型語言 有一個類型系統&#xff0c;不…

011_異常、泛型和集合框架

異常、泛型和集合框架 異常Java的異常體系異常的作用 自定義異常異常的處理方案異常的兩種處理方式 泛型泛型類泛型接口泛型方法、通配符和上下限泛型支持的類型 集合框架集合體系結構Collection Collection集合Collection的遍歷方式認識并發修改異常問題解決并發修改異常問題的…

Kubernetes 集群搭建(三):使用dashboard用戶界面(需要訪問外網獲取yaml)

&#xff08;一&#xff09;簡介 K8s Dashboard是Kubernetes提供的一種基于Web的用戶界面工具&#xff0c;用于可視化地管理和監控Kubernetes集群 主要功能&#xff1a; 資源查看與管理&#xff1a; 查看Kubernetes集群中的各種資源&#xff0c;如節點、Pod、服務、部署等。 對…

【數據挖掘】嶺回歸(Ridge Regression)和線性回歸(Linear Regression)對比實驗

這是一個非常實用的 嶺回歸&#xff08;Ridge Regression&#xff09;和線性回歸&#xff08;Linear Regression&#xff09;對比實驗&#xff0c;使用了 scikit-learn 中的 California Housing 數據集 來預測房價。 &#x1f4e6; 第一步&#xff1a;導入必要的庫 import num…

大疆無人機系列知識

目錄 知識 開發者文檔 &#xff08;上云&#xff09; 無人機的應用 知識 大疆行業無人機接入音視頻平臺協議詳解_大疆無人機 視頻流-CSDN博客 開發者文檔 &#xff08;上云&#xff09; 上云API 無人機的應用 【大疆無人機地圖測繪技術學習&#xff1a;高精度、高效率的…

CNN注意力機制的進化史:深度解析10種注意力模塊如何重塑卷積神經網絡

&#x1f31f; 引言&#xff1a;注意力為何改變CNN的命運&#xff1f; 就像人類視覺會優先聚焦于重要信息&#xff0c;深度學習模型也需要"學會看重點"。從2018年SENet首提通道注意力&#xff0c;到2024年SSCA探索空間-通道協同效應&#xff0c;注意力機制正成為CNN…

Linux/樹莓派網絡配置、遠程登錄與圖形界面訪問實驗

一.準備工作 1.修改網絡適配器&#xff08;選擇本機網卡&#xff09; 2.創建一個新的用戶。 3.使用新用戶登錄&#xff0c;使用ip a指令查看IP&#xff08;現代 Linux 發行版&#xff08;如 Ubuntu、Debian、CentOS、Fedora 等&#xff09;&#xff09;。 通過sudo arp-sca…

Python----TensorFlow(TensorFlow介紹,安裝,主要模塊,高級功能)

一、TensorFlow TensorFlow 是由谷歌大腦團隊于 2015 年推出的開源機器學習框架。作為深度學習的第二代系統&#xff0c;TensorFlow 支持多種編程語言&#xff0c;包括 Python、C、Java 和 Go&#xff0c;廣泛應用于 CNN、RNN 和 GAN 等深度學習算法。 TensorFlow 可以…

【動態規劃】 深入動態規劃 回文子串問題

文章目錄 前言例題一、回文子串二、 最長回文子串三、回文串分割IV四、分割回文串II五、最長回文子序列六、讓字符串成為回文串的最小插入次數 結語 前言 那么&#xff0c;什么是動態規劃中的回文子串問題呢&#xff1f; 動態規劃中的回文子串問題是一個經典的字符串處理問題。…

lodash庫介紹(一個現代JavaScript實用工具庫,提供模塊化、性能優化和額外功能)JavaScript庫(防抖、節流、函數柯里化)JS庫

https://www.lodashjs.com/ 文章目錄 Lodash庫全解析簡介核心優勢一致性API模塊化設計性能優化 常用功能分類數組操作對象操作函數增強 高級應用場景數據轉換鏈函數組合 性能考量大數據集處理 最佳實踐按需引入利用FP模塊 結語 Lodash庫全解析 簡介 Lodash是一個現代JavaScri…

Spring MVC 國際化機制詳解(MessageSource 接口體系)

Spring MVC 國際化機制詳解&#xff08;MessageSource 接口體系&#xff09; 1. 核心接口與實現類詳解 接口/類名描述功能特性適用場景MessageSource核心接口&#xff0c;定義消息解析能力支持參數化消息&#xff08;如{0}占位符&#xff09;所有國際化場景的基礎接口Resource…

PyTorch張量范數計算終極指南:從基礎到高階實戰

在深度學習領域&#xff0c;張量范數計算是模型正則化、梯度裁剪、特征歸一化的核心技術。本文將以20代碼實例&#xff0c;深度剖析torch.norm的9大核心用法&#xff0c;并揭示其在Transformer模型中的關鍵應用場景。 &#x1f680; 快速入門&#xff08;5分鐘掌握核心操作&…

榮耀90 GT信息

外觀設計 屏幕&#xff1a;采用 6.7 英寸 AMOLED 榮耀綠洲護眼屏&#xff0c;超窄邊框設計&#xff0c;其上邊框 1.6mm&#xff0c;左右黑邊 1.25mm&#xff0c;屏占較高&#xff0c;帶來更廣闊的視覺體驗。屏幕還支持 120Hz 自由刷新率&#xff0c;可根據使用場景自動切換刷新…

【Java中級】11章、枚舉 - java引用數據類型,枚舉介紹、快速入門,了解枚舉類的基本使用方式【1】

文章內容&#xff1a; 自定義實現枚舉enum關鍵字實現枚舉 ??內容涉及枚舉的定義&#xff0c;快速入門&#xff0c;注意事項和小題鞏固知識點 &#x1f308; 跟著B站一位老師學習的內部類內容&#xff0c;現寫這篇文章為學習內部類的小伙伴提供思路支持&#xff0c;希望可以一…

局域網訪問 Redis 方法

局域網訪問 Redis 方法 默認情況下&#xff0c;Redis 只允許本機 (127.0.0.1) 訪問。如果你想讓局域網中的其他設備訪問 Redis&#xff0c;需要 修改 Redis 配置&#xff0c;并確保 防火墻放行端口。 方法 1&#xff1a;修改 Redis 配置 1. 修改 redis.conf&#xff08;或 me…

如何應對客戶頻繁變更需求

如何應對客戶頻繁變更需求&#xff1f;要點包括&#xff1a; 快速響應、深入溝通、靈活規劃、過程記錄、風險管控。這些策略既能降低項目失控風險&#xff0c;也能幫助團隊在變動環境中保持高效率。其中深入溝通尤為關鍵&#xff0c;它不僅能夠讓團隊第一時間了解客戶意圖&…

Set 集合

默認情況下&#xff0c; Scala 使用的是不可變集合&#xff0c; 如果你想使用可變集合&#xff0c; 需要引用 scala.collection.mutable.Set Set 默認是不可變集合&#xff0c;數據無序 數據不可重復 遍歷集合 創建可變集合 mutable.Set 打印集合 集合添加元素 向集合中…

最新 OpenHarmony 系統一二級目錄整理

我們在學習 OpenHarmony 的時候&#xff0c;如果對系統的目錄結構了解&#xff0c;那么無疑會提升自己對 OpenHarmony 更深層次的認識。 于是就有了今天的整理。 首先在此之前&#xff0c;我們要獲取源碼 獲取源碼的方式 OpenHarmony 主干代碼獲取 方式一&#xff08;推薦&am…

STL常用容器整理

STL常用容器操作整理 STL常用容器操作整理&#xff08;string/vector/set/map&#xff09;一、string&#xff08;字符串&#xff09;構造函數元素訪問修改操作容量操作子串與查找 二、vector&#xff08;動態數組&#xff09;構造函數元素訪問修改操作容量操作 三、set&#x…