首页 > 其他 > 详细

BZOJ4347 POI2016Nim z utrudnieniem(博弈+动态规划)

时间:2018-10-27 17:59:42      阅读:141      评论:0      收藏:0      [点我收藏+]

  由nim游戏的结论,显然等价于去掉一些数使剩下的数异或和为0。

  暴力的dp比较显然,设f[i][j][k]为前i堆移走j堆(模意义下)后异或和为k的方案数。注意到总石子数量不超过1e7,按ai从小到大排序,这样k的枚举范围就不会超过2ai,于是复杂度O(md)。

  注意空间卡的非常紧,连滚动都开不下,改为留下的有j堆(模意义下)可能比较方便,存一下j=d-1时的数组,对j=1~d-1倒序转移完后再特判j=0即可。

#include<iostream> 
#include<cstdio>
#include<cmath>
#include<cstdlib>
#include<cstring>
#include<algorithm>
using namespace std;
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;
}
#define N 500010
#define P 1000000007
int n,m,a[N],u[N<<1],f[10][1<<20],tmp[1<<20];
inline void inc(int &x,int y){x+=y;if (x>=P) x-=P;}
int main()
{
#ifndef ONLINE_JUDGE
    freopen("bzoj4347.in","r",stdin);
    freopen("bzoj4347.out","w",stdout);
    const char LL[]="%I64d\n";
#else
    const char LL[]="%lld\n";
#endif
    n=read(),m=read();
    for (int i=1;i<=n;i++) a[i]=read();
    sort(a+1,a+n+1);
    int t=1;
    for (int i=1;i<=1000000;i++)
    {
        if (t<i) t=t<<1|1;
        u[i]=t;
    }
    f[0][0]=1;
    for (int i=1;i<=n;i++)
    {
        memcpy(tmp,f[m-1],u[a[i]]+1<<2);
        for (int j=m-1;j>=1;j--)
            for (int k=0;k<=u[a[i]];k++)
            inc(f[j][k],f[j-1][k^a[i]]);
        for (int k=0;k<=u[a[i]];k++)
        inc(f[0][k],tmp[k^a[i]]);
    }
    cout<<(f[n%m][0]-(n%m==0)+P)%P;
    return 0;
}

 

BZOJ4347 POI2016Nim z utrudnieniem(博弈+动态规划)

原文:https://www.cnblogs.com/Gloid/p/9862313.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!