码迷,mamicode.com
首页 > 其他好文 > 详细

10月10日|9日考试

时间:2017-10-10 17:48:53      阅读:484      评论:0      收藏:0      [点我收藏+]

标签:++   main   big   输出   names   ==   最大值   题解   节点   

big
【题目描述】
你需要在[0,2^n)中选一个整数 x,接着把 x 依次异或 m 个整数
a1~am。
在你选出 x 后,你的对手需要选择恰好一个时刻(刚选完数时、
2x
异或一些数后或是最后)
,将 x 变为(⌊ n ⌋ + 2x) mod 2 n 。
2
你想使 x 最后尽量大,而你的对手会使 x 最后尽量小。
你需要求出 x 最后的最大值,以及得到最大值的初值数量。
【输入数据】
第一行两个整数 n,m。第二行 m 个整数 a1~am
【输出数据】
第一行输出一个整数,表示 x 最后的最大值。
第二行输出一个整数,表示得到最大值的初值数量。
第一个数正确得 6 分,两个数都正确再得 4 分。
【样例输入】
2 3
1 2 3
【样例输出】
1
2
【样例解释】
x=0 时得到 0,x=1 时得到 1,x=2 时得到 1,x=3 时得到 0。【数据范围】
对于 20%的数据,n<=10,m<=100。
对于 40%的数据,n<=10,m<=1000。
对于另外 20%的数据,n<=30,m<=10。
对于 100%的数据 n<=30,m<=100000,0<=ai<2^n。

 

题解:

  看到2^n次方,首先想到2进制分解+tire树,然后考场上还发现了一个规律,(⌊ n ⌋ + 2x) mod 2^n就是将二进制位向前移动一位,然后超过2^n的向0位补齐,但想到这就不会了,我以为要tire树上写一个很难的dp,没想到一个转化。

  我们枚举一个断点x,当然我们暴力做是异或两个数,但是我们可以把前x个数,每个数都按照上面所讲的向前移位,因为位运算不会相互影响,那么先异或这前x个数之后进行转化,和异或这转化后的前x个数是一样的,所以我们用转化过的前x个数,异或后面的m-x个数就得到我们正真想异或的数。因为有m个断点,所以我们就有m个数。

  把m个树构在trie树上,那么考虑遍历这棵tire,剩下的就是简单贪心了,如果当前节点有两个儿子,那么对手一定会让你得不到这一位的1,如果只有一个,那么你就选和这个儿子不同的就行了。

 

代码:

#include <cstdio>
#include <iostream>
#include <algorithm>
#include <cstring>
#include <cmath>
#include <iostream>
#define MAXN 3500010
using namespace std;
int ch[MAXN][2],t[MAXN],a[MAXN],sum[MAXN],ans[MAXN];
int n,m,num=1,num2=0;

void insert(int x){
    for(int i=1;i<=n;i++) t[i]=(x>>(n-i)&1);
    int now=1;
    for(int i=1;i<=n;i++){
        if(!ch[now][t[i]]) ch[now][t[i]]=++num;
        now=ch[now][t[i]];
    }
}

void dfs(int now,int shu,int dep){
    if(dep<0){
        ans[++num2]=shu;
        return;
    }
    if(!ch[now][0]&&ch[now][1]) dfs(ch[now][1],shu^(1<<dep),dep-1);
    else if(!ch[now][1]&&ch[now][0]) dfs(ch[now][0],shu^(1<<dep),dep-1);
    else{
        dfs(ch[now][0],shu,dep-1);
        dfs(ch[now][1],shu,dep-1);
    }
}

bool cmp(int x,int y){
    return x>y;
}

int b[MAXN];

int main()
{
    scanf("%d%d",&n,&m);
    for(int i=1;i<=m;i++) scanf("%d",&a[i]);
    for(int i=m;i>=1;i--) sum[i]=sum[i+1]^a[i];
    int x=0;
    for(int i=0;i<=m;i++){
        x=x^((2*a[i])/(1<<n)+2*a[i])%(1<<n);
        b[i]=x^sum[i+1];
        insert(x^sum[i+1]);
    }
    dfs(1,0,n-1);
    sort(ans+1,ans+num2+1,cmp);
    int cnt=0;
    for(int i=1;i<=num2;i++){
        if(ans[i]==ans[1]) cnt++;
    }
    printf("%d\n%d",ans[1],cnt);
    return 0;
}

 

10月10日|9日考试

标签:++   main   big   输出   names   ==   最大值   题解   节点   

原文地址:http://www.cnblogs.com/renjianshige/p/7646779.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!