Golang实现0-1背包问题方案(动态规划-贪心算法-回溯-分支定界)
创始人
2025-05-31 22:07:02
0

Golang实现0-1背包问题方案(动态规划-贪心算法-回溯-分支定界)

算法实现说明

  • 动态规划算法时间复杂度较低,能够求解较大规模的问题,但空间复杂度较高,不适用于数据量较大的问题。
  • 贪心算法时间复杂度较低,能够求解较大规模的问题,但不能保证求得的解是最优解。
  • 回溯算法能够求解较小规模的问题,但时间复杂度较高,不适用于数据量较大的问题。
  • 分支定界算法能够求解较小规模的问题,但时间复杂度较高,不适用于数据量较大的问题。

0-1背包问题说明

0-1背包问题是一个经典的组合优化问题,其问题描述如下:

有一个容量为 CCC 的背包,和 nnn 个物品,每个物品有重量 wiw_iwi​ 和价值 viv_ivi​,现在需要从这 nnn 个物品中选择一些物品放入背包中,使得这些物品的总重量不超过 CCC,且这些物品的总价值最大。

0-1背包问题是一个 NP 完全问题,因此不存在多项式时间复杂度的算法能够求解该问题。但是,我们可以使用多种算法来求解该问题,包括动态规划、贪心、回溯、分支定界算法等。

动态规划算法

动态规划算法是求解0-1背包问题的一种常用算法,其时间复杂度较低,能够求解较大规模的问题。动态规划算法的基本思想是将原问题分解为若干个子问题,先求解子问题,再由子问题的解得到原问题的解。

动态规划算法求解0-1背包问题的时间复杂度为 O(n∗c)O(n*c)O(n∗c),空间复杂度为 O(n∗c)O(n*c)O(n∗c),其中 nnn 表示物品的数量,ccc 表示背包的容量。

/*** 动态规划求解0-1背包问题* @param w 物品重量* @param v 物品价值* @param c 背包容量* @return 背包能装下的最大价值* 时间复杂度:O(n*c)* 空间复杂度:O(n*c)* 优点:时间复杂度较低,能够求解较大规模的问题* 缺点:空间复杂度较高,不适用于数据量较大的问题*/
func knapsack(w []int, v []int, c int) int {n := len(w)dp := make([][]int, n+1)for i := 0; i <= n; i++ {dp[i] = make([]int, c+1)}for i := 1; i <= n; i++ {for j := 1; j <= c; j++ {if j < w[i-1] {// 背包容量不足,不能装入第i个物品dp[i][j] = dp[i-1][j]} else {// 能装入第i个物品dp[i][j] = max(dp[i-1][j], dp[i-1][j-w[i-1]]+v[i-1])}}}// 返回背包能装下的最大价值return dp[n][c]
}

贪心算法

贪心算法是求解0-1背包问题的一种常用算法,其时间复杂度较低,能够求解较大规模的问题。贪心算法的基本思想是每次选择当前最优的解,直到得到最终的解。

贪心算法求解0-1背包问题的时间复杂度为 O(n∗log(n))O(n*log(n))O(n∗log(n)),空间复杂度为 O(n)O(n)O(n)。

 */
func greedyKnapsack(w []int, v []int, c int) int {n := len(w)index := make([]int, n)for i := 0; i < n; i++ {index[i] = i}temp := make([]int, n)for i := 0; i < n; i++ {temp[i] = v[i] * w[i]}for i := 0; i < n; i++ {for j := i + 1; j < n; j++ {if temp[i] < temp[j] {t := temp[i]temp[i] = temp[j]temp[j] = tt = index[i]index[i] = index[j]index[j] = t}}}maxValue := 0for i := 0; i < n; i++ {if w[index[i]] <= c {maxValue += v[index[i]]c -= w[index[i]]} else {maxValue += int(float64(v[index[i]])/float64(w[index[i]])*float64(c))break}}return maxValue
}

回溯算法

回溯算法是求解0-1背包问题的一种常用算法,其能够求解较小规模的问题。回溯算法的基本思想是搜索所有可能的解,找到最优解。

回溯算法求解0-1背包问题的时间复杂度为 O(2n)O(2^n)O(2n),空间复杂度为 O(n)O(n)O(n)。

/*** 回溯算法求解0-1背包问题* @param w 物品重量* @param v 物品价值* @param c 背包容量* @return 背包能装下的最大价值* 时间复杂度:O(2^n)* 空间复杂度:O(n)* 优点:能够求解较小规模的问题* 缺点:时间复杂度较高,不适用于数据量较大的问题*/
func backtrackKnapsack(w []int, v []int, c int) int {n := len(w)maxValue := 0for i := 0; i < (1 << n); i++ {currentWeight := 0currentValue := 0for j := 0; j < n; j++ {if (i & (1 << j)) != 0 {currentWeight += w[j]currentValue += v[j]}}if currentWeight <= c && currentValue > maxValue {maxValue = currentValue}}return maxValue
}

分支定界算法

分支定界算法是求解0-1背包问题的一种常用算法,其能够求解较小规模的问题。分支定界算法的基本思想是搜索所有可能的解,但是在搜索过程中,通过一些限制条件,剪枝掉一些不可能成为最优解的分支,从而减少搜索的时间。

分支定界算法求解0-1背包问题的时间复杂度为 O(2n)O(2^n)O(2n),空间复杂度为 O(n)O(n)O(n)。

/*** 分支定界算法求解0-1背包问题* @param w 物品重量* @param v 物品价值* @param c 背包容量* @return 背包能装下的最大价值* 时间复杂度:O(2^n)* 空间复杂度:O(n)* 优点:能够求解较小规模的问题* 缺点:时间复杂度较高,不适用于数据量较大的问题*/
func branchBoundKnapsack(w []int, v []int, c int) int {n := len(w)index := make([]int, n)for i := 0; i < n; i++ {index[i] = i}temp := make([]int, n)for i := 0; i < n; i++ {temp[i] = v[i] * w[i]}for i := 0; i < n; i++ {for j := i + 1; j < n; j++ {if temp[i] < temp[j] {t := temp[i]temp[i] = temp[j]temp[j] = tt = index[i]index[i] = index[j]index[j] = t}}}maxValue := 0currentValue := 0currentWeight := 0i := 0for i >= 0 {if i == n {if currentValue > maxValue {maxValue = currentValue}i--} else if currentWeight+w[index[i]] <= c {currentWeight += w[index[i]]currentValue += v[index[i]]i++} else {i--}if i >= n || i < 0 || index[i] >= n || currentWeight+w[index[i]] > c || currentValue+(c-currentWeight)*v[index[i]]/w[index[i]] <= maxValue {i--}}return maxValue
}

算法汇总代码

package mainimport ("fmt""time"
)func main() {// 物品重量w := []int{2, 4, 6, 8, 9, 2, 4, 6, 8, 9, 2, 4, 6, 8, 9, 2, 4, 6, 8, 9}// 物品价值v := []int{1, 3, 5, 7, 9, 1, 3, 5, 7, 9, 1, 3, 5, 7, 9, 1, 3, 5, 7, 9}// 背包容量1c1 := 10// 背包容量2c2 := 100start1 := time.Now()result1 := knapsack(w, v, c1)end1 := time.Now()start2 := time.Now()result2 := knapsack(w, v, c2)end2 := time.Now()fmt.Printf("当背包容量为10时,动态规划算法的结果为:%d,执行时间为:%v\n", result1, end1.Sub(start1))fmt.Printf("当背包容量为10时,贪心算法的结果为:%d\n", greedyKnapsack(w, v, c1))fmt.Printf("当背包容量为10时,回溯算法的结果为:%d\n", backtrackKnapsack(w, v, c1))fmt.Printf("当背包容量为10时,分支定界算法的结果为:%d\n", branchBoundKnapsack(w, v, c1))fmt.Printf("当背包容量为100时,动态规划算法的结果为:%d,执行时间为:%v\n", result2, end2.Sub(start2))fmt.Printf("当背包容量为100时,贪心算法的结果为:%d\n", greedyKnapsack(w, v, c2))fmt.Printf("当背包容量为100时,回溯算法的结果为:%d\n", backtrackKnapsack(w, v, c2))fmt.Printf("当背包容量为100时,分支定界算法的结果为:%d\n", branchBoundKnapsack(w, v, c2))
}/*** 动态规划求解0-1背包问题* @param w 物品重量* @param v 物品价值* @param c 背包容量* @return 背包能装下的最大价值* 时间复杂度:O(n*c)* 空间复杂度:O(n*c)* 优点:时间复杂度较低,能够求解较大规模的问题* 缺点:空间复杂度较高,不适用于数据量较大的问题*/
func knapsack(w []int, v []int, c int) int {n := len(w)dp := make([][]int, n+1)for i := 0; i <= n; i++ {dp[i] = make([]int, c+1)}for i := 1; i <= n; i++ {for j := 1; j <= c; j++ {if j < w[i-1] {// 背包容量不足,不能装入第i个物品dp[i][j] = dp[i-1][j]} else {// 能装入第i个物品dp[i][j] = max(dp[i-1][j], dp[i-1][j-w[i-1]]+v[i-1])}}}// 返回背包能装下的最大价值return dp[n][c]
}/*** 贪心算法求解0-1背包问题* @param w 物品重量* @param v 物品价值* @param c 背包容量* @return 背包能装下的最大价值* 时间复杂度:O(n*log(n))* 空间复杂度:O(n)* 优点:时间复杂度较低,能够求解较大规模的问题* 缺点:不能保证求得的是最优解,只能得到一个近似解*/
func greedyKnapsack(w []int, v []int, c int) int {n := len(w)index := make([]int, n)for i := 0; i < n; i++ {index[i] = i}temp := make([]int, n)for i := 0; i < n; i++ {temp[i] = v[i] * w[i]}for i := 0; i < n; i++ {for j := i + 1; j < n; j++ {if temp[i] < temp[j] {t := temp[i]temp[i] = temp[j]temp[j] = tt = index[i]index[i] = index[j]index[j] = t}}}maxValue := 0for i := 0; i < n; i++ {if w[index[i]] <= c {maxValue += v[index[i]]c -= w[index[i]]} else {maxValue += int(float64(v[index[i]])/float64(w[index[i]])*float64(c))break}}return maxValue
}/*** 回溯算法求解0-1背包问题* @param w 物品重量* @param v 物品价值* @param c 背包容量* @return 背包能装下的最大价值* 时间复杂度:O(2^n)* 空间复杂度:O(n)* 优点:能够求解较小规模的问题* 缺点:时间复杂度较高,不适用于数据量较大的问题*/
func backtrackKnapsack(w []int, v []int, c int) int {n := len(w)maxValue := 0for i := 0; i < (1 << n); i++ {currentWeight := 0currentValue := 0for j := 0; j < n; j++ {if (i & (1 << j)) != 0 {currentWeight += w[j]currentValue += v[j]}}if currentWeight <= c && currentValue > maxValue {maxValue = currentValue}}return maxValue
}/*** 分支定界算法求解0-1背包问题* @param w 物品重量* @param v 物品价值* @param c 背包容量* @return 背包能装下的最大价值* 时间复杂度:O(2^n)* 空间复杂度:O(n)* 优点:能够求解较小规模的问题* 缺点:时间复杂度较高,不适用于数据量较大的问题*/
func branchBoundKnapsack(w []int, v []int, c int) int {n := len(w)index := make([]int, n)for i := 0; i < n; i++ {index[i] = i}temp := make([]int, n)for i := 0; i < n; i++ {temp[i] = v[i] * w[i]}for i := 0; i < n; i++ {for j := i + 1; j < n; j++ {if temp[i] < temp[j] {t := temp[i]temp[i] = temp[j]temp[j] = tt = index[i]index[i] = index[j]index[j] = t}}}maxValue := 0currentValue := 0currentWeight := 0i := 0for i >= 0 {if i == n {if currentValue > maxValue {maxValue = currentValue}i--} else if currentWeight+w[index[i]] <= c {currentWeight += w[index[i]]currentValue += v[index[i]]i++} else {i--}if i >= n || i < 0 || index[i] >= n || currentWeight+w[index[i]] > c || currentValue+(c-currentWeight)*v[index[i]]/w[index[i]] <= maxValue {i--}}return maxValue
}

相关内容

热门资讯

龚玥版祝酒歌 龚玥版祝酒歌 龚... 龚玥版祝酒歌 民歌贺新春祝酒歌歌词美酒飘香啊歌声飞朋友啊请你干一杯请你干一杯胜利的十月永难忘杯中洒满...
公司宴会祝酒词 祝酒范文 公司... 公司宴会祝酒词 祝酒范文尊敬的各位领导、各位来宾,女士们、先生们:  今晚,我们欢聚一堂,共同祝贺中...
刘和刚版祝酒歌 刘和刚版祝酒歌... 刘和刚版祝酒歌!欢乐!中国行!现场!国语!
关牧村现场版祝酒歌 关牧村现场... 关牧村现场版祝酒歌祝酒歌演唱:关牧村专集:新阿瓦古丽美酒飘香啊歌声飞朋友啊请你干一杯请你干一杯胜利的...
二老生日快乐祝酒诗 二老生日快... 二老生日快乐祝酒诗小时勤勉老无忧旭日经年光由足恭贺佳期人增寿祝酒举杯庆吉祥二月初二龙抬头老当益壮今胜...
刀郎版祝酒歌 刀郎版祝酒歌 祝... 刀郎版祝酒歌祝酒歌美酒飘香啊歌声飞朋友啊请你干一杯请你干一杯胜利的十月永难忘杯中洒满幸福泪来来来来来...
家庭聚会祝酒词 家庭聚会祝酒词... ◆家庭聚会祝酒辞 敬爱的长辈们: 晚上好! 新春共饮团圆酒,家家幸福加新年。在今天这个辞旧迎新的日子...
同事聚会祝酒词 同事聚会祝酒词... 同事聚会祝酒词劝酒者起身敬酒,被劝者会说:"屁股一抬,喝了重来",意让劝酒者再喝一个。此时劝酒者应对...
战友聚会祝酒词 战友聚会祝酒词... 战友聚会祝酒词各位战友、各位兄长,各位小弟大家好。今天是我军建军81周年,在这特殊的节日里,我衷心的...
李光羲版祝酒歌 李光羲版祝酒歌... 李光羲版祝酒歌 美酒飘香啊歌声飞朋友啊请你干一杯请你干一杯胜利的十月永难忘杯中洒满幸福泪来来来来来来...
爆笑GIF图:有这样的损友,原... 1.有这样的损友,原谅我笑点低哈哈2.少年的出行工具也太奇葩了!没系安全带,扣分!3.热气腾腾烤红薯...
最新或2023(历届)鼠年单位... 值此辞旧迎新之际,我们谨代表x(单位名),向各位员工、各位朋友,各位客户,致以节日的问候和新年的祝福...
笑话GIF:我是不是被骗了,总... 不准笑,谁还没有个童心未泯的时候。能看出这张图的诡异之处吗?这种笑容,只有女人看到喜欢的那个人,才会...
大学同学聚会祝酒词 大学同学聚... 大学同学聚会祝酒词女士们、先生们、敬爱的老师、亲爱的同学们: 大家中午好!今天,我们山东师范大...
爸妈是真爱,你只是意外,致那些... 宠物千万别和乌龟一块儿养!不然。。。红衣服的男的你是诚实的哪位大神解释一下为什么狗眼睛瞬间就绿了??...
国庆放假安排最新或2023(历...   学院办公室关于最新或2023(历届)国庆节放假安排的通知  各教学单位,各室、部、中心、馆:  ...
最新或2023(历届)关于印发...  近日,中共中央印发了《中国共产党廉洁自律准则》(以下简称《准则》),并发出通知,要求各地区各部门认...
朱元璋当时处于底层,为什么能起... 嗨又和大家见面了,今天趣历史小编带来了一篇关于朱元璋的文章,希望你们喜欢。对于任何一个开创帝业的帝王...
唐朝时期的四大美食介绍 当时的... 最近有很多古代和如今的对比,说古代的小官员的物质条件享受其实还不如现在的一个小市民。比如说古代不能随...
揭秘:朱元璋为什么承认元朝为正... 嗨又和大家见面了,今天趣历史小编带来了一篇关于朱元璋的文章,希望你们喜欢。元朝是我们中国历史上的一个...