zoukankan      html  css  js  c++  java
  • BZOJ5467 PKUWC2018Slay the Spire(动态规划)

      即求所有情况的最大伤害之和。容易发现应该先打强化牌,至少打一张攻击牌。同样显然的是强化牌和攻击牌都应该按从大到小的顺序打。进一步可以发现,只要还有强化牌,就应该使用(当然至少留一次攻击的机会)。

      于是将强化牌和攻击牌各自从大到小排序。显然可以将其分开考虑。对强化牌,设f[i][j]为前i张牌抽到j张并打出的强化倍数之和,则显然有f[i][j]=f[i-1][j]+f[i-1][j-1]·w[i]。这样就搞定了强化牌可以打完的情况。同时设g[i]为抽i张打出k-1张的强化倍数之和,dp过程中通过f数组计算,注意避免重复。对于攻击牌也进行类似dp。然后枚举两种牌各抽了几张合并一下答案即可。注意细节。

    #include<iostream> 
    #include<cstdio>
    #include<cmath>
    #include<cstdlib>
    #include<cstring>
    #include<algorithm>
    using namespace std;
    #define ll long long
    #define N 3010
    #define P 998244353
    char getc(){char c=getchar();while ((c<'A'||c>'Z')&&(c<'a'||c>'z')&&(c<'0'||c>'9')) c=getchar();return c;}
    int gcd(int n,int m){return m==0?n:gcd(m,n%m);}
    int read()
    {
        int x=0,f=1;char c=getchar();
        while (c<'0'||c>'9') {if (c=='-') f=-1;c=getchar();}
        while (c>='0'&&c<='9') x=(x<<1)+(x<<3)+(c^48),c=getchar();
        return x*f;
    }
    int T,n,m,k,a[N],b[N],f[2][N][N],g[2][N],C[N][N],h[N];
    int main()
    {
    #ifndef ONLINE_JUDGE
        freopen("bzoj5467.in","r",stdin);
        freopen("bzoj5467.out","w",stdout);
        const char LL[]="%I64d
    ";
    #else
        const char LL[]="%lld
    ";
    #endif
        T=read();
        while (T--)
        {
            n=read(),m=read(),k=read();int lim=min(k-1,n);
            C[0][0]=1;
            for (int i=1;i<=n;i++)
            {
                C[i][0]=C[i][i]=1;
                for (int j=1;j<n;j++)
                C[i][j]=(C[i-1][j-1]+C[i-1][j])%P;
            }
            memset(f,0,sizeof(f));memset(g,0,sizeof(g));memset(h,0,sizeof(h));
            for (int i=1;i<=n;i++) a[i]=read();
            for (int i=1;i<=n;i++) b[i]=read();
            sort(a+1,a+n+1),reverse(a+1,a+n+1);
            sort(b+1,b+n+1),reverse(b+1,b+n+1);
            f[0][0][0]=1;
            for (int i=1;i<=n;i++)
            {
                f[0][i][0]=1;
                for (int j=1;j<=lim;j++)
                f[0][i][j]=(f[0][i-1][j]+1ll*f[0][i-1][j-1]*a[i])%P;
                if (lim==0) for (int j=0;j<=n;j++) g[0][j]=C[n][j];
                else
                for (int j=lim;j<=n;j++)
                g[0][j]=(g[0][j]+1ll*f[0][i-1][lim-1]*a[i]%P*C[n-i][j-lim])%P;
            }
            for (int i=1;i<=n;i++)
            {
                for (int j=1;j<=m;j++)
                f[1][i][j]=(f[1][i-1][j]+f[1][i-1][j-1]+1ll*b[i]*C[i-1][j-1])%P;
                for (int j=m-k+1;j<=n;j++)
                g[1][j]=(g[1][j]+(f[1][i-1][j-(m-k)-1]+1ll*b[i]*C[i-1][j-(m-k)-1])%P*C[n-i][m-k])%P;
            }
            for (int i=1;i<=min(n,m);i++)
                for (int j=1;j<=n;j++)
                h[i]=(h[i]+1ll*b[j]*C[n-j][i-1])%P;
            /*for (int i=0;i<=n;i++) cout<<f[0][n][i]<<' ';cout<<endl;
            for (int i=0;i<=n;i++) cout<<g[0][i]<<' ';cout<<endl;
            for (int i=0;i<=n;i++) cout<<f[1][n][i]<<' ';cout<<endl;
            for (int i=0;i<=n;i++) cout<<g[1][i]<<' ';cout<<endl;
            for (int i=0;i<=n;i++) cout<<h[i]<<' ';cout<<endl;*/
            int ans=0;
            for (int i=max(0,m-n);i<=min(n,m);i++)
            if (i<=lim) ans=(ans+1ll*f[0][n][i]*g[1][m-i])%P;//抽了i张强化牌 全部出完 剩余m-i张攻击牌 选m-k张不出
            else ans=(ans+1ll*g[0][i]*h[m-i])%P;//抽了i张强化牌 选lim张出 剩余m-i张攻击牌 出1张
            printf("%d
    ",ans);
        }
        return 0;
    }
     
  • 相关阅读:
    阿里云短信支付微信支付
    python 阿里云短信群发推送
    python twilio 短信群发 知识留存
    python爬虫 发送定时气象预报
    python beautifulsoup爬虫
    strongswan
    Docker进入主流,PaaS大有可为(转)
    Python 网页爬虫 & 文本处理 & 科学计算 & 机器学习 & 数据挖掘兵器谱(转)
    在IT网站上少花些时间
    Python 代码性能优化技巧(转)
  • 原文地址:https://www.cnblogs.com/Gloid/p/10136466.html
Copyright © 2011-2022 走看看