Jurassic Remains UVALive - 2965(中途相遇法)

题目链接

https://vjudge.net/problem/UVALive-2965

题意

给定n个由大写字母组成的字符串。选尽可能多的字符串,使得每个大写字母都出现偶数次

分析

要求每个字母出现偶数次,我们不需要去关心每个字母最终出现了多少次,只需要知道其出现的次数是奇数还是偶数。这样,我们就可以用一个比较高效的方法来判定选取的字符串是否符合“使得每个大写字母都出现偶数次”的要求。一个字母出现了偶数次,那么对这偶数次个字母进行异或后得到的值一定为0.

那么要怎么进行字符串之间的异或呢?
我们可以把一个字符串映射为一个整数。
比如将“ABEZ”映射为二进制表示为”10000000000000000000010011“的整数。
用这些个整数之间的异或来替代字符串之间的异或。

接下来要解决的问题就是”如何选取字符串“
先考虑朴素算法:
对每个字符串去枚举选取、不选取两种状态。这2^n种可能中,状态为选取的字符串的数目最多且其异或值为0的情况即可所求。而题干给出n最大为24,O(2^n)级别的算法肯定超时。
然后考虑优化或者其他算法:
当枚举的维数比较多时,一般考虑部分枚举。比如要枚举区间,我们可以考虑只枚举终点,然后在枚举终点的时候通过边枚举边递推来确定起点。但是这个题的维数有24之多 ,这个优化策略不太好实行。
如果我们不用for循环来进行枚举,而是用建立一颗深度优先搜索树(dfs树)形式来枚举呢?我们知道在深度比较深的dfs树中的搜索优化有种策略叫做双向搜索。在起点和终点明确的情况下,同时从起点和终点取进行搜索,这样搜索深度就减半了。
中途相遇法:
对前n/2个字符串进行枚举,将枚举结果映射到map中。然后对剩下的字符串进行枚举,看所能得到的所有异或值在map中是否存在。

代码

#include <cstdio>
#include <iostream>
#include <map>

using namespace std;

const int maxn=100;
int a[maxn]; /*a[i]的值的二进制表示 等于第i个字符串的
             的“二进制”(从最低位到最高位依此表示A~Z
            该字母有奇数个则该位为1,否则为0*/
char s[1005]; //临时存放字符串
map<int,int> table;
/*table的第二位表示怎么选字符串,如为7,二进制表示为111,
则选第1、2、3号字符串;table的第一位表示该种选法对应的
异或值*/

//x的二进制表示中‘1’的个数
int bitcount(int x)
{
    return x==0?0:(x&1)+bitcount(x>>1);
}
int main()
{
    int n;
    while(~scanf("%d",&n))
    {
        for(int i=0;i<n;i++)
        {
            scanf("%s",s);
            a[i]=0;
            for(int j=0;s[j]!='\0';j++)
                a[i]^=1<<(s[j]-'A');
        }
        table.clear();
        int n1=n/2,n2=n-n1;
        for(int i=0;i<(1<<n1);i++) //2^n1种选取方法,二进制位上的值为1表示选取该二进制位对应的字符串
        {
            int x=0;
            for(int j=0;j<n1;j++)if(i & (1<<j))
                x^=a[j];
            if(!table.count(x) || bitcount(table[x])<bitcount(i))
                table[x]=i;
        }
        int ans=0;
        for(int i=0;i<(1<<n2);i++)
        {
            int x=0;
            for(int j=0;j<n2;j++)if(i & (1<<j))
                x^=a[j+n1];
            if(table.count(x)&&bitcount(table[x])+bitcount(i)>bitcount(ans))
                ans=(i<<n1)^table[x];
        }
        printf("%d\n",bitcount(ans));
        int k=0;
        while(ans)
        {
            k++;
            if(ans&1) printf("%d ",k);
            ans>>=1;
        }
        printf("\n");
       /* for(int i=0;i<n;i++)
            if(ans & (1<<i))
            printf("%d ",i+1);
        printf("\n");*/
    }
    return 0;

}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值