<?xml version="1.0" encoding="utf-8" standalone="yes"?><rss version="2.0" xmlns:atom="http://www.w3.org/2005/Atom"><channel><title>Go on Lost Temple</title><link>https://cloudcold.ai/tags/go/</link><description>Recent content in Go on Lost Temple</description><generator>Hugo</generator><language>en</language><lastBuildDate>Thu, 28 Sep 2017 22:27:21 +0000</lastBuildDate><atom:link href="https://cloudcold.ai/tags/go/index.xml" rel="self" type="application/rss+xml"/><item><title>背包问题学习</title><link>https://cloudcold.ai/posts/2017-09-28-%E8%83%8C%E5%8C%85%E9%97%AE%E9%A2%98%E5%AD%A6%E4%B9%A0/</link><pubDate>Thu, 28 Sep 2017 22:27:21 +0000</pubDate><guid>https://cloudcold.ai/posts/2017-09-28-%E8%83%8C%E5%8C%85%E9%97%AE%E9%A2%98%E5%AD%A6%E4%B9%A0/</guid><description>&lt;p&gt;今天学习了下背包问题，也就是传说中的NP问题，屌屌的。我学习的是最简单的背包问题。&lt;/p&gt;
&lt;p&gt;打个简单的比方，假设你找到一片宝藏，这时候你有一个背包，这个背包是有容量限制的，最多只能装10公斤的东西。&lt;/p&gt;
&lt;p&gt;然后宝藏中有各种宝石，黄金，和价值连城的珍珠，可惜的是，每件宝物都有其对应的价值和重量，这个时候问题就来了，你怎么选择，能够使得你背包里面装的宝物价值最大化呢？&lt;/p&gt;
&lt;p&gt;假设现在宝藏中只有一个宝石，其价值是5，重量是3，那么你的选择是将宝石装进背包，这个时候，背包所承载的物品的价值是5，背包剩余的重量还剩7.&lt;/p&gt;
&lt;p&gt;那么再假设下，现在宝藏中有两块宝石，宝石A价值是5，重量是3，宝石B的价值是6，重量是6. 那么你的选择当然是宝石A和宝石B一起放进去。那么这个时候，包包所承载的宝石价值就是11，重量是9. 那么假设宝石B的价值是6，但重量是8，那么就不能把A和B都放在背包了，因为两者重量加起来超过了10.所以要求价值最大化，就只能放宝石B。&lt;/p&gt;
&lt;p&gt;还可以依次往下看三块宝石，四块宝石，五块宝石。。。的选择情况。&lt;/p&gt;
&lt;p&gt;其实我这里没有讲的特别学院派，讲状态和状态转移方程。但是我的理解就是这样的。于是我照着这个思路，以及网上的源码，边理解边写了一遍，然后又用go去实现了这个算法。&lt;/p&gt;
&lt;p&gt;这个问题可以用DP来解，也可以用MIP来解。不过对于我来说，用DP来解就已经很受用了，MIP太高端，我也摸不太透。&lt;/p&gt;
&lt;p&gt;下面就解释下我的代码。&lt;/p&gt;
&lt;div class="highlight"&gt;&lt;pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-text-size-adjust:none;"&gt;&lt;code class="language-python" data-lang="python"&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#66d9ef"&gt;def&lt;/span&gt; &lt;span style="color:#a6e22e"&gt;pack_result&lt;/span&gt;():
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#e6db74"&gt;&amp;#34;&amp;#34;&amp;#34;背包最大容量是10 总共4件物品，价值和重量分别如下
&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#e6db74"&gt; Knapsack Max weight : W = 10 
&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#e6db74"&gt; Total items : N = 4
&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#e6db74"&gt; Values of items : v= [10, 40, 30, 50]
&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#e6db74"&gt; Weight of items : w = [5, 4, 6, 3]
&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#e6db74"&gt;
&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#e6db74"&gt; return：拿到最大价值，和背包的剩余容量
&lt;/span&gt;&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#e6db74"&gt; &amp;#34;&amp;#34;&amp;#34;&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; V &lt;span style="color:#f92672"&gt;=&lt;/span&gt; [&lt;span style="color:#ae81ff"&gt;10&lt;/span&gt;,&lt;span style="color:#ae81ff"&gt;40&lt;/span&gt;,&lt;span style="color:#ae81ff"&gt;30&lt;/span&gt;,&lt;span style="color:#ae81ff"&gt;50&lt;/span&gt;] &lt;span style="color:#75715e"&gt;# 每个物品对应的价值&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; W &lt;span style="color:#f92672"&gt;=&lt;/span&gt; [&lt;span style="color:#ae81ff"&gt;5&lt;/span&gt;,&lt;span style="color:#ae81ff"&gt;2&lt;/span&gt;,&lt;span style="color:#ae81ff"&gt;8&lt;/span&gt;,&lt;span style="color:#ae81ff"&gt;7&lt;/span&gt;] &lt;span style="color:#75715e"&gt;#每个物品的重量&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; MAX &lt;span style="color:#f92672"&gt;=&lt;/span&gt; &lt;span style="color:#ae81ff"&gt;9&lt;/span&gt; &lt;span style="color:#75715e"&gt;#背包的最大载重量&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; print getValue(W,V,MAX,&lt;span style="color:#ae81ff"&gt;4&lt;/span&gt;) &lt;span style="color:#75715e"&gt;#i=4,就代表有4件物品&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#66d9ef"&gt;def&lt;/span&gt; &lt;span style="color:#a6e22e"&gt;getValue&lt;/span&gt;(W,V,MAX,i): &lt;span style="color:#75715e"&gt;#定义一个函数&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#66d9ef"&gt;if&lt;/span&gt; i&lt;span style="color:#f92672"&gt;&amp;gt;&lt;/span&gt;&lt;span style="color:#ae81ff"&gt;1&lt;/span&gt;: 
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#75715e"&gt;#不放第i件物品最大价值&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; DoNotPutThe_indexi_goods_in_bag_Value,\
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; DoNotPutThe_indexi_goods_in_bag_Weight &lt;span style="color:#f92672"&gt;=&lt;/span&gt; getValue(W,V,MAX,i&lt;span style="color:#f92672"&gt;-&lt;/span&gt;&lt;span style="color:#ae81ff"&gt;1&lt;/span&gt;)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#75715e"&gt;#如果第i件物品的重量大于背包最大容量&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#66d9ef"&gt;if&lt;/span&gt; W[i&lt;span style="color:#f92672"&gt;-&lt;/span&gt;&lt;span style="color:#ae81ff"&gt;1&lt;/span&gt;] &lt;span style="color:#f92672"&gt;&amp;gt;&lt;/span&gt; MAX: &lt;span style="color:#75715e"&gt;# 如果这个地方为真，那就不能放i进去了，因为放i进去，背包就装不下了，既然不放i，那就是返回上面不放i的值，&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#66d9ef"&gt;return&lt;/span&gt; DoNotPutThe_indexi_goods_in_bag_Value,DoNotPutThe_indexi_goods_in_bag_Weight
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#66d9ef"&gt;else&lt;/span&gt;: &lt;span style="color:#75715e"&gt;#如果第i件物品的重量小于背包最大容量，那就可以把i放进去了。那把i放进去了后，包包剩下的重量就是max减去i的重量，&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; changed_Value,changed_Weight &lt;span style="color:#f92672"&gt;=&lt;/span&gt; getValue(W,V,MAX&lt;span style="color:#f92672"&gt;-&lt;/span&gt;W[i&lt;span style="color:#f92672"&gt;-&lt;/span&gt;&lt;span style="color:#ae81ff"&gt;1&lt;/span&gt;],i&lt;span style="color:#f92672"&gt;-&lt;/span&gt;&lt;span style="color:#ae81ff"&gt;1&lt;/span&gt;)
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#66d9ef"&gt;if&lt;/span&gt; changed_Value &lt;span style="color:#f92672"&gt;+&lt;/span&gt; V[i&lt;span style="color:#f92672"&gt;-&lt;/span&gt;&lt;span style="color:#ae81ff"&gt;1&lt;/span&gt;] &lt;span style="color:#f92672"&gt;&amp;gt;&lt;/span&gt; DoNotPutThe_indexi_goods_in_bag_Value:
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#66d9ef"&gt;return&lt;/span&gt; changed_Value &lt;span style="color:#f92672"&gt;+&lt;/span&gt; V[i&lt;span style="color:#f92672"&gt;-&lt;/span&gt;&lt;span style="color:#ae81ff"&gt;1&lt;/span&gt;],changed_Weight
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#66d9ef"&gt;else&lt;/span&gt;:
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#66d9ef"&gt;return&lt;/span&gt; DoNotPutThe_indexi_goods_in_bag_Value,DoNotPutThe_indexi_goods_in_bag_Weight
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#75715e"&gt;#可以先看else下面的，比较容易懂，意思就是如果只有一件物品，并且大于背包的重量，那么背包所承载的价值就为0，背包剩余容量就是没装东西的容量MAX&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#66d9ef"&gt;else&lt;/span&gt;:
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#66d9ef"&gt;if&lt;/span&gt; W[&lt;span style="color:#ae81ff"&gt;0&lt;/span&gt;] &lt;span style="color:#f92672"&gt;&amp;gt;&lt;/span&gt; MAX:
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#66d9ef"&gt;return&lt;/span&gt; &lt;span style="color:#ae81ff"&gt;0&lt;/span&gt;,MAX
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#75715e"&gt;# 如果这件物品小于背包的最大容量，那么就装进背包，那么背包所承载的价值就是这件物品的价值，背包剩余容量就是最大容量减去这件物品的重量&lt;/span&gt;
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#66d9ef"&gt;else&lt;/span&gt;:
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; &lt;span style="color:#66d9ef"&gt;return&lt;/span&gt; V[&lt;span style="color:#ae81ff"&gt;0&lt;/span&gt;],MAX &lt;span style="color:#f92672"&gt;-&lt;/span&gt; W[&lt;span style="color:#ae81ff"&gt;0&lt;/span&gt;]
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt;&lt;span style="color:#66d9ef"&gt;if&lt;/span&gt; __name__ &lt;span style="color:#f92672"&gt;==&lt;/span&gt; &lt;span style="color:#e6db74"&gt;&amp;#39;__main__&amp;#39;&lt;/span&gt;:
&lt;/span&gt;&lt;/span&gt;&lt;span style="display:flex;"&gt;&lt;span&gt; pack_result()
&lt;/span&gt;&lt;/span&gt;&lt;/code&gt;&lt;/pre&gt;&lt;/div&gt;&lt;p&gt;理解了原理，然后我有用Go实现了一版：&lt;/p&gt;</description></item></channel></rss>