[Ceoi2010]Pin

#2012. [Ceoi2010]Pin

Online Judge:Bzoj-2012

Label:容斥,STL

题目描述

给出N(2<=N<=50000)个长度为4的字符串,问有且仅有D(1<=D<=4)处不相同的字符串有几对

输入格式

第1行: N,D
以下N行每行一个字符串

输出格式

一个数:有多少对有且仅有处不相同的字符串。

样例

输入

4 2
0000
a010
0202
a0e2

输出

3

题解:

n的数据范围为50000,暴力(O(N^2*D))铁T。

先转化一下问题,它问有且仅有d处不同的对数,那我们就将其转化为有且仅有4-d处相同的对数。这两个问题是完全等价的。我们考虑继续简化问题,上面这个问题难在“有且仅有”,发现可以把它转化为“有4-d处相同的对数”,然后利用容斥解决问题。

于是问题就变的很水了,由于只字符串长度固定为4,我们用二进制x表示当前考虑的是哪几位(比如((1001)_2表示只考虑字符串的第1和第4位)),先预处理出二进制表示的数(x∈[0,15])所含的1的个数(sz1[x])

接下来枚举当前数x,然后再枚举n个字符串,根据要考虑的位进行哈希(map搞一下),然后进行计数,将考虑x这几位的贡献加入(rc[sz1[x]])((rc[i]表示有i出相同的字符串的对数)),最后根据组合数容斥一下,得出答案。

代码如下,详见注释↘

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
char s[50000][5];
int sz1[20],rc[5],n,d;
map<ll,int>mp;
int main(){
	scanf("%d%d",&n,&d);
	for(int i=1;i<=n;i++)scanf("%s",s[i]);
	for(int i=0;i<(1<<4);i++)sz1[i]=sz1[i>>1]+(i&1);//预处理含1个数
	for(int i=0;i<=15;i++){
		ll cnt=0;mp.clear();
		for(int j=1;j<=n;j++){
			ll id=0;//哈希,由于可能出现的可见字符貌似(255???)个左右,下面乘*260
			for(int k=0;k<4;k++)if(i&(1<<k))id=id*260+s[j][k];
			cnt+=mp[id],mp[id]++;
		}
		rc[sz1[i]]+=cnt;
	}
    //根据组合数容斥
	rc[3]-=rc[4]*4;//c(4,3)
	rc[2]-=rc[4]*6+rc[3]*3;//c(4,2),c(3,2)
	rc[1]-=rc[4]*4+rc[3]*3+rc[2]*2;
	rc[0]-=rc[4]+rc[3]+rc[2]+rc[1];
	printf("%d",rc[4-d]);
}
原文地址:https://www.cnblogs.com/Tieechal/p/11226407.html