母函数题目

hdu 1028 Ignatius and the Princess III 母函数模板题

http://acm.hdu.edu.cn/showproblem.php?pid=1028

题意:

整数的拆分。

母函数学习http://www.wutianqi.com/?p=596

View Code
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
#include <queue>
#include <stack>
#include <set>
#include <map>
#include <string>

#define CL(a,num) memset((a),(num),sizeof(a))
#define iabs(x)  ((x) > 0 ? (x) : -(x))
#define Min(a , b) ((a) < (b) ? (a) : (b))
#define Max(a , b) ((a) > (b) ? (a) : (b))

#define ll __int64
#define inf 0x7f7f7f7f
#define MOD 100000007
#define lc l,m,rt<<1
#define rc m + 1,r,rt<<1|1
#define pi acos(-1.0)
#define test puts("<------------------->")
#define maxn 100007
#define M 100007
#define N 127
using namespace std;
//freopen("din.txt","r",stdin);

int c1[N],c2[N];

void init(int n){
    int i,j,k;
    for (i = 0; i <= n; ++i){
        c1[i] = 1;//乘完后的系数
        c2[i] = 0;//乘的过程中的系数
    }

    for (i = 2; i <= n; ++i){
        //总共有n个括号,从第2个起每一个括号都要和前面那一个括号相乘
    //所以可以忽略第一个括号 
        for (j = 0; j <= n; ++j){//j代表最前面这个大括号的项数 
            for (k = 0; k + j <= n; k += i){//在大括号后面,x都是以i方递增的 
                c2[k + j] += c1[j];//这里就是大括号后面的括号与前面相乘的计算 
            }
        }
        for (j = 0; j <= n; ++j){
            c1[j] = c2[j];
            c2[j] = 0;
        }
    }
}
int main(){
    int n;
    init(120);
    while (~scanf("%d",&n)){
        printf("%d\n",c1[n]);
    }
    return 0;
}

 

hdu 1398 Square Coins

http://acm.hdu.edu.cn/showproblem.php?pid=1398

题意:

整数拆分的题目,只不过后边的括号里面的差值变成了i^2吧了。。

View Code
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
#include <queue>
#include <stack>
#include <set>
#include <map>
#include <string>

#define CL(a,num) memset((a),(num),sizeof(a))
#define iabs(x)  ((x) > 0 ? (x) : -(x))
#define Min(a , b) ((a) < (b) ? (a) : (b))
#define Max(a , b) ((a) > (b) ? (a) : (b))

#define ll __int64
#define inf 0x7f7f7f7f
#define MOD 100000007
#define lc l,m,rt<<1
#define rc m + 1,r,rt<<1|1
#define pi acos(-1.0)
#define test puts("<------------------->")
#define maxn 100007
#define M 100007
#define N 327
using namespace std;
//freopen("din.txt","r",stdin);

int c1[N],c2[N];
int pow2[320];

void init(int n){

    int i,j,k;
    for (i = 1; i <= 300; ++i){
        pow2[i] = i*i;
    }
    for (i = 0; i <= n; ++i){
        c1[i] = 1;
        c2[i] = 0;
    }

    for (i = 2; i <= n; ++i){
        for (j = 0; j <= n; ++j){
            for (k = 0; k + j <= n; k += pow2[i]){
                c2[k + j] += c1[j];
            }
        }
        for (j = 0; j <= n; ++j){
            c1[j] = c2[j];
            c2[j] = 0;
        }
    }
}
int main(){
    int n;
    init(300);
    while (~scanf("%d",&n)){
        if (!n) break;
        printf("%d\n",c1[n]);
    }
    return 0;
}

 

hdu 1085 Holding Bin-Laden Captive!

http://acm.hdu.edu.cn/showproblem.php?pid=1085

题意:

给出三种硬币,价值分别为1,2,5并给出他们的数量num1,num2,num5问他们不能够组合出来的的最小价值;

思路:

前边我们介绍的都是要么是每种数量为1个或者无限个,而这里给出了确定的数量。而且询问的是不能组合出来额最小值。所以有点绕,但思路还是没有改变。

我们只要在进行乘法运算时动态的增加括号里的项数即可。最后从小到大枚举一下就好。。

View Code
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
#include <queue>
#include <stack>
#include <set>
#include <map>
#include <string>

#define CL(a,num) memset((a),(num),sizeof(a))
#define iabs(x)  ((x) > 0 ? (x) : -(x))
#define Min(a , b) ((a) < (b) ? (a) : (b))
#define Max(a , b) ((a) > (b) ? (a) : (b))

#define ll __int64
#define inf 0x7f7f7f7f
#define MOD 100000007
#define lc l,m,rt<<1
#define rc m + 1,r,rt<<1|1
#define pi acos(-1.0)
#define test puts("<------------------->")
#define maxn 100007
#define M 100007
#define N 8007
using namespace std;
//freopen("din.txt","r",stdin);

int c1[N],c2[N];

int main(){
    int i,j,k;
    int num1,num2,num5;
    while (~scanf("%d%d%d",&num1,&num2,&num5)){
        if (!num1 && !num2 && !num5) break;
        int MAX = num1 + num2*2 + num5*5;//本身所能表示的最大数
        for (i = 0; i <= MAX; ++i){
            c1[i] = 0;
            c2[i] = 0;
        }
        for (i = 0; i <= num1; ++i) c1[i] = 1;//初始化第一个括号里的项
        //成第二个括号里的项
        for (j = 0; j <= num1; ++j){
            for (k = 0; k <= num2*2; k += 2){
                c2[k + j] += c1[j];
            }
        }
        for (j = 0; j <= num1 + num2*2; ++j){//动态的添加了
            c1[j] = c2[j];
            c2[j] = 0;
        }
        for (j = 0; j <= num1 + num2*2; ++j){
            for (k = 0; k <= num5*5; k += 5){
                c2[k + j] += c1[j];
            }
        }
        for (j = 0; j <= MAX; ++j){
            c1[j] = c2[j];
            c2[j] = 0;
        }
        for (i = 1; i <= MAX; ++i){
            if (!c1[i]) break;
        }
        printf("%d\n",i);
    }
    return 0;
}

 hdu Big Event in HDU

http://acm.hdu.edu.cn/showproblem.php?pid=1171

题意:

给出N中不同的设备,分别给出他们的价值以及数量,将这些设备分为两组,尽量使这两组的价值相等。输出各自的价值,若不相等大的在前边;

思路:

和上边一样。。。不多讲;

View Code
#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cmath>
#include <queue>
#include <stack>
#include <set>
#include <map>
#include <string>

#define CL(a,num) memset((a),(num),sizeof(a))
#define iabs(x)  ((x) > 0 ? (x) : -(x))
#define Min(a , b) ((a) < (b) ? (a) : (b))
#define Max(a , b) ((a) > (b) ? (a) : (b))

#define ll __int64
#define inf 0x7f7f7f7f
#define MOD 100000007
#define lc l,m,rt<<1
#define rc m + 1,r,rt<<1|1
#define pi acos(-1.0)
#define test puts("<------------------->")
#define maxn 100007
#define M 100007
#define N 250007
using namespace std;
//freopen("din.txt","r",stdin);

int c1[N],c2[N];

int num[55],val[55];

int main(){
   // freopen("din.txt","r",stdin);
    int n,i,j,k;
    while (~scanf("%d",&n)){
        if (n < 0) break;
        int MAX = 0;
        for (i = 0; i < n; ++i){
            scanf("%d%d",&val[i],&num[i]);
            MAX += (val[i]*num[i]);
        }
        for (i = 0; i <= MAX; ++i){
            c1[i] = c2[i] = 0;
        }
        int len = val[0]*num[0];
        for (i = 0; i <= len; i += val[0]){
            c1[i] = 1;
        }
        for (i = 1; i < n; ++i){
            for (j = 0; j <= len; ++j){
                for (k = 0; k <= val[i]*num[i]; k += val[i]){
                    c2[k + j] += c1[j];
                }
            }
            len += val[i]*num[i];
            for (j = 0; j <= len; ++j){
                c1[j] = c2[j];
                c2[j] = 0;
            }
        }
        int ave = MAX/2;
        for (i = ave; i >= 0; --i){
            if (c1[i]) break;
        }
        int A = i;
        int B = MAX - A;
        printf("%d %d\n",max(A,B),min(A,B));
    }
    return 0;
}
posted @ 2012-10-06 19:51  E_star  阅读(256)  评论(0编辑  收藏  举报