题目链接
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;
}