漫画:什么是“贪心算法”?如何求解“部分背包问题”?

55f43af29db7cd1da51ab80200675298.png

70cab8a5388ec9149eb7ffba6d782d59.png

—————  第二天  —————

519436f08b9c0e20fff3c7d2c94a73a0.png

605a85b37c015997441baa6651067c89.png

809dac7718c8e2f209fb3e4d9a4b769b.png

068437e9f1fabd6d53310e3924c7cb82.png

4ef6b096bdfd5e8ea45006d8fd968ab2.png

d862b85331e49a3cb78955e0dcdde275.png

6c12e9707beb9213aba9e18516f3c868.png

de8633087a622b014c46354c738e0b65.png

e90f4657b7afe38ab52e89199494674c.png

————————————

d79dc0406d881ec76d231e24feaca188.png

c716bd42e6cc7f1c20bbdfb171f4a1d4.png

df8216e66b8e86eb5b51ef5054dc6086.png

. . . . . . . .

4ace0e611a9f5f8433ef7f5c652a43f3.png

94472a5652d3586062859489473e4f83.png

ea34cd51e8ec00a6c858c17ed459caa6.png

f2bf67e81729c7627cdaa9546fd6ec30.png

75813df1b4e6f080ed3572079e3e2d4a.png

我们回到刚才的题目当中,假设背包的容量是10,有5个商品可供选择,每个商品的价值和重量如图所示:

3fb821358aa1e66b89045cf546561483.png

045ca4e0ccf7093496d683d068d46d71.png

995bedde85688913714172d7e31d176e.png

7a8d04e140f60d911aad41efbebcb261.png

让我们来计算一下每件物品的性价比,其结果如下:

76b763b7a8ec2f48e157f62d4e4fb8dd.png

毫无疑问,此时性价比最高的是物品4,我们把物品4放入背包当中,背包剩余的容量是8:

f36468c00fd1800bbb2b5e289771b671.png

292dcf8cae1a90a5b2217a9a393f3614.png

af0654649988202fa636a3b47b826cdf.png

bf5ff5987793a467f6f0931907635f6a.png

我们选择物品1放入背包,背包剩余的容量是4:

d77125a91e9b86c1791ab12a59073c87.png

d68a2c331e656e01f8e43f45fb6f9885.png

53d5c49dca7c367dd98c2d574ceea756.png

于是,我们选择0.8份的物品5放入背包,背包剩余的容量为0:

181ba8e92f0b2dbd047f87932fefee6c.png

2e7173a414d2c288a8859a3ad8cb414b.png

b26fbeaa04b0431686f40b2e499506c1.png

1338bdf963d788dec79a33fbc8398933.png

a8b50467d7b27398471d7d951ea567e4.png

0ea3f5c16051c55a4653bae35501ca9a.png

public static void main(String[] args) {
        int capacity = 10;
        int[] weights = {4,6,3,2,5};
        int[] values = {9,3,1,6,5};
        System.out.println("背包最大价值:" + getHighestValue(capacity, weights, values));
    }

    public static double getHighestValue(int capacity, int[] weights,int[] values){

        //创建物品列表并按照性价比倒序
        List<Item> itemList = new ArrayList<>();
        for(int i=0;i<weights.length;i++){
            itemList.add(new Item(weights[i], values[i]));
        }
        itemList = itemList.stream().sorted(Comparator.comparing(Item::getRatio).reversed()).collect(Collectors.toList());

        //背包剩余容量
        int restCapacity = capacity;
        //当前背包物品的最大价值
        double highestValue = 0;

        //按照性价比从高到低选择物品
        for(Item item : itemList){
            if(item.weight <= restCapacity){
                highestValue += item.value;
                restCapacity -= item.weight;
            }else{
                //背包装不下完整物品时,选择该件物品的一部分
                highestValue += (double)restCapacity/(double)item.weight * item.value;
                break;
            }
        }

        return highestValue;
    }

    static class Item {
        private int weight;
        private int value;
        //物品的性价比
        private double ratio;

        public Item (int weight, int value){
            this.weight = weight;
            this.value = value;
            this.ratio = (double)value / (double)weight;
        }

        public double getRatio() {
            return ratio;
        }
    }

在这段代码当中,我们借助了静态内部类Item,从而更方便地记录性价比、获取重量和价值信息、按性价比排序。

7e49f41353e95418680067b432dbccea.png

eb7284a1121079a208a50d6c4e8ee131.png

630d6cd84b8a565e25aa5329fa03d93c.png

af4bb2835e30654b4c92b7e14e6168e8.png

仍然给定一个容量是10的背包,有如下三个物品可供选择:

b855e7198f16ded9bd966cd29eb2573d.png

这一次我们有个条件限制:只允许选择整个物品,不能选择物品的一部分。

如果按照贪心算法的思路,首先选择的是性价比最高的物品1,那么背包剩余容量是4,再也装不下其他物品,而此时的总价值是6:

00c1e3da1b850b404ba262063abded17.png

但这样的选择,真的能让总价值最大化吗?如果我们不选择物品1,选择物品2和物品3的话,剩余容量是0,总价值是7:

5f4e0523485e7d6e79c63eb9a50f8a20.png

显然,7>6,依靠贪心算法得出的结果,未必是全局最优解。

cb06952e9bd2e5f08126cd91c6878ff8.png

68206a1e224719c61f636464ec8d4b20.png

f363990c5356b02566fa720508e3d99d.png

漫画:什么是动态规划?(整合版)

7a0dc446aed0eddbe033e12cc7182628.png

猜你喜欢

转载自blog.csdn.net/bjweimengshu/article/details/120984292