整数划分

                        
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
整数划分问题是算法中的一个经典命题之一,有关这个问题的讲述在讲解到递归时基本都将涉及。所谓整数划分,是指把一个正整数n写成如下形式:</p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
       n=m1+m2+...+mi; (其中mi为正整数,并且1 <= mi <= n),则{m1,m2,...,mi}为n的一个划分。</p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
       如果{m1,m2,...,mi}中的最大值不超过m,即max(m1,m2,...,mi)<=m,则称它属于n的一个m划分。这里我们记n的m划分的个数为f(n,m);</p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
       例如但n=4时,他有5个划分,{4},{3,1},{2,2},{2,1,1},{1,1,1,1};</p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
       注意4=1+3 和 4=3+1被认为是同一个划分。</p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
       该问题是求出n的所有划分个数,即f(n, n)。下面我们考虑求f(n,m)的方法;</p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
 </p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
        <span style="line-height:19px;"><span style="line-height:1.8;color:#FF0000;">---------------------------------------------------------------------</span></span></p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
<span style="line-height:19px;"><span style="line-height:1.8;color:#FF0000;">                                           (一)递归法</span></span></p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
<span style="line-height:19px;"><span style="line-height:1.8;color:#FF0000;">        <span style="line-height:19px;"><span style="line-height:1.8;">---------------------------------------------------------------------</span></span></span></span></p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
       根据n和m的关系,考虑以下几种情况: </p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
       (1)当n=1时,不论m的值为多少(m>0),只有一种划分即{1};</p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
        (2)  当m=1时,不论n的值为多少,只有一种划分即n个1,{1,1,1,...,1};</p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
        (3)  当n=m时,根据划分中是否包含n,可以分为两种情况:</p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
              (a). 划分中包含n的情况,只有一个即{n};</p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
              (b). 划分中不包含n的情况,这时划分中最大的数字也一定比n小,即n的所有(n-1)划分。</p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
              因此 f(n,n) =1 + f(n,n-1);</p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
        (4) 当n<m时,由于划分中不可能出现负数,因此就相当于f(n,n);</p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
        (5) 但n>m时,根据划分中是否包含最大值m,可以分为两种情况:</p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
               (a). 划分中包含m的情况,即{m, {x1,x2,...xi}}, 其中{x1,x2,... xi} 的和为n-m,可能再次出现m,因此是(n-m)的m划分,因此这种划分</p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
                     个数为f(n-m, m);</p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
               (b). 划分中不包含m的情况,则划分中所有值都比m小,即n的(m-1)划分,个数为f(n,m-1);</p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
              因此 f(n, m) = f(n-m, m)+f(n,m-1);</p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
 </p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
         综合以上情况,我们可以看出,上面的结论具有递归定义特征,其中(1)和(2)属于回归条件,(3)和(4)属于特殊情况,将会转换为情况(5)。而情况(5)为通用情况,属于递推的方法,其本质主要是通过减小m以达到回归条件,从而解决问题。其递推表达式如下:</p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
         f(n, m)=       1;                                (n=1 or m=1)</p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
                            f(n, n);                         (n<m)</p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
                            1+ f(n, m-1);                (n=m)</p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
                            f(n-m,m)+f(n,m-1);       (n>m)</p>
<p style="margin:10px auto;font-family:Verdana, Arial, Helvetica, sans-serif;font-size:13px;line-height:23px;background-color:rgb(245,245,245);">
 </p><div class="dp-highlighter bg_cpp"><div class="bar"><div class="tools"><b>[cpp]</b> <a href="#" class="ViewSource" title="view plain" οnclick="dp.sh.Toolbar.Command('ViewSource',this);return false;">view plain</a><span class="tracking-ad" data-mod="popu_168"> <a href="#" class="CopyToClipboard" title="copy" οnclick="dp.sh.Toolbar.Command('CopyToClipboard',this);return false;">copy</a><div style="position: absolute; left: 242px; top: 1328px; width: 16px; height: 16px; z-index: 99;"><embed id="ZeroClipboardMovie_1" src="http://static.blog.csdn.net/scripts/ZeroClipboard/ZeroClipboard.swf" loop="false" menu="false" quality="best" bgcolor="#ffffff" name="ZeroClipboardMovie_1" allowscriptaccess="always" allowfullscreen="false" type="application/x-shockwave-flash" pluginspage="http://www.macromedia.com/go/getflashplayer" flashvars="id=1&width=16&height=16" wmode="transparent" width="16" height="16" align="middle"></div></span><span class="tracking-ad" data-mod="popu_169"> <a href="#" class="PrintSource" title="print" οnclick="dp.sh.Toolbar.Command('PrintSource',this);return false;">print</a></span><a href="#" class="About" title="?" οnclick="dp.sh.Toolbar.Command('About',this);return false;">?</a></div></div><ol start="1" class="dp-cpp"><li class="alt"><span><span>linux@ubuntu:~/workdir/ACM-ICPC$ cat IntDivide.c  </span></span></li><li class=""><span><span class="preprocessor">#include <stdio.h></span><span>  </span></span></li><li class="alt"><span>  </span></li><li class=""><span><span class="datatypes">int</span><span> IntDivide(</span><span class="datatypes">int</span><span> n, </span><span class="datatypes">int</span><span> m)  </span></span></li><li class="alt"><span>{  </span></li><li class=""><span>        <span class="keyword">if</span><span>((n<1)||(m<1))  </span></span></li><li class="alt"><span>                <span class="keyword">return</span><span> 0;  </span></span></li><li class=""><span>        <span class="keyword">if</span><span>((n==1)||(m==1))  </span></span></li><li class="alt"><span>                <span class="keyword">return</span><span> 1;  </span></span></li><li class=""><span>        <span class="keyword">if</span><span>(n<m)  </span></span></li><li class="alt"><span>                <span class="keyword">return</span><span> IntDivide(n,n);  </span></span></li><li class=""><span>        <span class="keyword">if</span><span>(n==m)  </span></span></li><li class="alt"><span>                <span class="keyword">return</span><span> IntDivide(n,n-1)+1;  </span></span></li><li class=""><span>        <span class="keyword">return</span><span> IntDivide(n-m,m)+IntDivide(n,m-1);  </span></span></li><li class="alt"><span>}  </span></li><li class=""><span>  </span></li><li class="alt"><span><span class="datatypes">int</span><span> main()  </span></span></li><li class=""><span>{  </span></li><li class="alt"><span>        <span class="datatypes">int</span><span> n=0,m=0;  </span></span></li><li class=""><span>        <span class="datatypes">int</span><span> result=0;  </span></span></li><li class="alt"><span>  </span></li><li class=""><span>        scanf(<span class="string">"%d %d"</span><span>,&n,&m);  </span></span></li><li class="alt"><span>  </span></li><li class=""><span>        result = IntDivide(n,m);  </span></li><li class="alt"><span>  </span></li><li class=""><span>        printf(<span class="string">"%d\n"</span><span>,result);  </span></span></li><li class="alt"><span>  </span></li><li class=""><span>        <span class="keyword">return</span><span> 0;  </span></span></li><li class="alt"><span>}  </span></li></ol></div><pre class="cpp" name="code" style="display: none;">linux@ubuntu:~/workdir/ACM-ICPC$ cat IntDivide.c
#include <stdio.h>

int IntDivide(int n, int m)
{
        if((n<1)||(m<1))
                return 0;
        if((n==1)||(m==1))
                return 1;
        if(n<m)
                return IntDivide(n,n);
        if(n==m)
                return IntDivide(n,n-1)+1;
        return IntDivide(n-m,m)+IntDivide(n,m-1);
}

int main()
{
        int n=0,m=0;
        int result=0;

        scanf("%d %d",&n,&m);

        result = IntDivide(n,m);

        printf("%d\n",result);

        return 0;
}
</pre><br>                    

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值