【动态规划】【排列组合】Wiseking (Wiseking.pas/c/cpp)

WISEKING
Wiseking.pas/c/cpp

   WISEKINGDOM 有N 个公主和M个侍卫暑假到了,公主们要去度假。为了公主的安全WISEKING 将让侍卫保护公主出行,并且每个公主至少需要两个侍卫;现在WISEKING

想知道一共有多少种分配方案;

   请你帮WISEKING 求出一共有多少种方案,并输出。

   输入:仅一行,两个数N,M;

   输出:方案数ANS;

样例1

  Wiseking.in

  3  5

  Wiseking.out

  0

样例2

  Wiseking.in

  2 7

  Wiseking.out

4

样例2说明

   第一个公主        第二个公主

2                                                            5

3                                                            4

4                                                            3

5                                                            2

对所有数据N<100,M<=500;

 

 

 

首先深搜是可以实现的,不过时间效率太低!

然后就是动规,我们用f[i][j]表示 i 个公主有 j 个护卫的方案总数,用样例举例

f[2][7]表示2个公主,7个护卫的方案总数,而第一个公主可以有2,3,4,5个公主,那么剩下的就是第二个公主的护卫,可以表示为

f[2][7]  <--  f[1][2]和f[1][5]
    <--  f[1][3]和f[1][4]
    <--  f[1][4]和f[1][3]
    <--  f[1][5]和f[1][5]

也就是f[i][j]我们就可以分解为f[1][k]和f[i-1][j-k],由于f[1][k]只可能有一种方案,所以根据加法和乘法原理,f[i][j]的方案数就可以表示为

f[i][j]=f[i-1][j-2]+f[i-1][j-3]+f[i-1][j-4]+...

这样就可以用递推解决,效率远远低于O(n*m*m)

然后几组极限数据就会发现 int 是存不下的,试着用unsigned long long,测完了发现仍然存不下(和long long的效果一样),就只得了60分

所以就要自己写高精度了

写完了一测,超时了。。。。。。后来分析频繁高精度效率一下子就低了!所以方程要简化

我们再来看看方程f[i][j]=f[i-1][j-2]+f[i-1][j-3]+f[i-1][j-4]+...

既然f[i][j]可以写成f[i-1][j-2]+f[i-1][j-3]+f[i-1][j-4]+....,那么同样f[i][j-1]也可以写成f[i-1][j-3]+f[i-1][j-4]+....

所以方程就可以写成f[i][j]=f[i-1][j-2]+f[i][j-1]

其实用到了排列组合的思想,然后写成高精度,就能AC了

C++ Code

/*
C++ Code
http://oijzh.cnblogs.com
By jiangzh
*/
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
using namespace std;
#define max(a,b) ((a)>(b)?(a):(b))

int n,m;
int f[110][510][100];//f[i][j]表示i个公主j个护卫的方案总数

void Plus(int *c,int *a,int *b)
{
    c[0]=max(a[0],b[0]);
    for(int i=1;i<=c[0];i++)
    {
        c[i]+=a[i]+b[i];
        if(c[i]>=10)
        {
            c[i+1]++;
            c[i]-=10;
        }
    }
    while(c[c[0]+1]>0)c[0]++;
}

void output(int *a)
{
    for(int i=a[0];i>0;i--)
        printf("%d",a[i]);
}

int main()
{
    freopen("Wiseking.in","r",stdin);
    freopen("Wiseking.out","w",stdout);
    scanf("%d%d",&n,&m);
    for(int i=1;i<=n;i++)for(int j=1;j<=m;j++)f[i][j][0]=1;
    for(int j=2;j<=m;j++)f[1][j][1]=1;
    for(int i=2;i<=n;i++)
        for(int j=1;j<=m;j++)
            Plus(f[i][j],f[i-1][j-2],f[i][j-1]);
    output(f[n][m]);
    return 0;
}

  

 

转载于:https://www.cnblogs.com/oijzh/archive/2012/10/30/2747131.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
#include #include #include #include #include #define N 100 int n=0; int Checktoseek(char name[]);//查找 void mainmenu();//菜单 void Register();//登记注册 void Showplayer();//显示当前人物数据 void Showallplayer();//显示所有人物数据 void Chooseopponent();//选择对手 void Chooseplayer(int i);//选择角色 void vs(int i,int j);//PK void Attack(int i,int j);//攻击 void recovery(int i,int j);//防御 void Energy(int i,int j);//能量 struct Kof { char name[20]; int Hp; int AP; int Dp; int Rp; }player[N]; void main() { srand(unsigned(time(NULL))); mainmenu(); } void login() { } void mainmenu() { while(1) { int x; printf("\n"); printf("\n"); printf(" ***************************************************************\n"); printf(" *** ***\n"); printf(" *** ***\n"); printf(" *** 欢迎进入Kof ***\n"); printf(" *** ***\n"); printf(" *** ***\n"); printf(" ***************************************************************\n"); printf("\t\t\t\t\t1----请先注册\n"); printf("\t\t\t\t\t2----查看玩家信息\n"); printf("\t\t\t\t\t3----查看自己信息\n"); printf("\t\t\t\t\t4----选择对手\n"); printf("\t\t\t\t\t5----结束\n"); printf("\n\n请选择菜单:"); scanf("%d",&x); if(x==5) break; switch(x) { case 1:Register();break; case 2:Showallplayer();break; case 3:Showplayer();break; case 4:Chooseopponent();break; } printf("\n\n\n\n\n按任意键继续:"); getch(); system("cls"); } } void Register() { char name[20]; printf("输入名称:"); fflush(stdin); gets(name); if(Checktoseek(name)==-1) { strcpy(player[n].name,name); player[n].Hp=rand()%1000; player[n].AP=rand()%(100+1)+50; player[n].Dp=rand()%(10-5+2)+5; player[n].Rp=rand()%(60+2)+60; n++; printf("人物创建成功\n"); } else { Register(); } } void Showplayer() { char name[20]; printf("角色名:"); fflush(stdin); gets(name); int i=Checktoseek(name); if(i!=-1) { printf("人物名称:%s\n",player[i].name); printf("Hp:%d\n",player[i].Hp); printf("Ap:%d\n",player[i].AP); printf("Dp:%d\n",player[i].Dp); printf("Rp:%d\n",player[i].Rp); } else { printf("请重新输入角色名"); Showplayer(); } } void Showallplayer() { printf("%-10s%-10s%-10s%-10s%-10s\n","Name","Hp","Ap","Dp","Rp"); for(int i=0;i<n;i++) { printf("%-10s%-10d%-10d%-10d%-10d\n",player[i].name,player[i].Hp,player[i].AP,player[i].Dp,player[i].Rp); } } void Chooseopponent() { char name[20]; int i; Showallplayer(); printf("你选择PK谁:"); fflush(stdin); gets(name); i=Checktoseek(name); if(i!=-1) { Chooseplayer(i); } else { printf("你选择PK的人物找不到?\n"); Chooseopponent(); } } int Checktoseek(char name[]) { for(int i=0;i0 && player[j].Hp>0) { printf("第%d局***************\n",count); if(count%2==0) { Attack(i,j); recovery(i,j); Energy(i,j); } else { Attack(j,i); recovery(i,j); Energy(i,j); } count++; _sleep(3344); } if(player[i].Hp>0) { printf("%s Victory \n",player[i].name); } else { printf("%s Winner \n",player[j].name); } printf("******************************\n"); } void Attack(int i,int j) { int x; int s; x=rand()%5; switch(x) { case 0: printf("\n%s发出波动伤害%s\n",player[i].name,player[j].name); s=rand()%100; player[j].Hp-=s; printf("%s受到%d点伤害\n\n",player[j].name,s); break; case 1: printf("\n%s使用连招攻击%s\n",player[i].name,player[j].name); s=rand()%100*2; player[j].Hp-=s; printf("%s受到%d点伤害\n\n",player[j].name,s); break; case 2: printf("\n%s释放超杀技\n",player[i].name); s=(rand()%100)*4; player[j].Hp-=s; printf("%s受到%d点伤害\n\n",player[j].name,s); break; } } void recovery(int i,int j) { int s; int x; s=rand()%3; switch(s) { case 0: printf("\n%s防御%s的攻击\n",player[i].name,player[j].name); x=rand()%(10-5+2)+5; player[i].Dp==x; printf("%s抵挡%d点伤害\n\n",player[i].name,x); break; case 1: printf("\n%s防御%s的攻击\n",player[j].name,player[i].name); x=rand()%(10-5+2)+5; player[j].Dp==x; printf("%s抵挡%d点伤害\n\n",player[j].name,x); break; } } void Energy(int i,int j) { int x; int s; x=rand()%6; switch(x) { case 0: printf("\n%s大怒,使用超必杀技\n",player[i].name); s=(rand()%100)*6; player[j].Hp-=s; printf("%s受到%d点伤害\n\n",player[j].name,s); break; case 1: printf("\n%s大怒,使用超必杀技\n",player[j].name); s=(rand()%100)*6; player[i].Hp-=s; printf("%s受到%d点伤害\n\n",player[i].name,s); break; } }
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值