zoukankan      html  css  js  c++  java
  • hlg1541集合划分【01背包】

    集合划分
    Time Limit: 1000 MS Memory Limit: 65535 K
    Total Submit: 44(17 users) Total Accepted: 22(16 users) Rating:  Special Judge: No
    Description
    对于从1到N (1 <= N <= 39) 的连续整数集合,能划分成两个子集合,且保证每个集合的数字和是相等的。举个例子,如果N=3,对于{1,2,3}能划分成两个子集合,每个子集合的所有数字和是相等的:
    {3} 和 {1,2}
    这是唯一一种分法(交换集合位置被认为是同一种划分方案,因此不会增加划分方案总数) 如果N=7,有四种方法能划分集合{1,2,3,4,5,6,7},每一种分法的子集合各数字和是相等的:
    {1,6,7} 和 {2,3,4,5} {注 1+6+7=2+3+4+5}
    {2,5,7} 和 {1,3,4,6}
    {3,4,7} 和 {1,2,5,6}
    {1,2,4,7} 和 {3,5,6}
    给出N,你的程序应该输出划分方案总数,如果不存在这样的划分方案,则输出0。
    Input

    有多组测试数据。

    对于每组测试数据,输入一个整数n。

    Output
    对于每组测试数据,输出划分方案总数,如果不存在则输出0。
    Sample Input
    7
    Sample Output
    4

    分析:

    每个数的组合方式翻了一倍  左边一遍右边一遍  求完之后除以2就好了

     代码:

     1 #include <iostream>
     2 #include <cstdio>
     3 #include <cstring>
     4 using namespace std;
     5 
     6 const int maxn = 781;
     7 
     8 long long dp[maxn];
     9 int n;
    10 int sum;
    11 
    12 void DP() {
    13     memset(dp, 0, sizeof(dp));
    14     dp[0] = 1;
    15     for(int i = 1; i <= n; i++) {
    16         for(int j = sum; j >= i; j--) {
    17             dp[j] += dp[j - i];
    18         }
    19     }
    20     printf("%lld
    ", dp[sum] / 2);
    21 }
    22 
    23 int main() {
    24     for(n = 1; n <= 39; n++) {
    25         sum = ( 1 + n ) * n / 2;
    26         if(sum & 1) {
    27             puts("0");
    28         } else {
    29             sum /= 2;
    30             DP();
    31         }
    32     }
    33     return 0;
    34 }
    View Code
  • 相关阅读:
    dnn
    DATAGRID学习
    在.net下的换行符
    treeview
    《25项最优时间管理工具与技巧》
    vim常用操作
    【Google给毕业生的忠告】
    MySQL的安装、使用及权限管理
    各种国际化标准组织
    ubuntu thunderbird 邮箱 163 配置 不能发送问题
  • 原文地址:https://www.cnblogs.com/zhanzhao/p/4145714.html
Copyright © 2011-2022 走看看