【poj1733】 Parity game

http://poj.org/problem?id=1733 (题目链接)

题意

  一个由0,1组成的序列,每次给出一段区间的奇偶,问哪一条信息不合法。

Solution

  并查集。

  题目中序列的长度有很大,单纯搜索一定会TLE。

  我们用s[i]表示前i个数的前缀和,那么a b even意味着s[b]和s[a-1]的奇偶性相同。a b odd意味着s[b]与s[a-1]的奇偶性不同。于是我们根据奇偶性的不同,用并查集依次处理他们之间的关系。当某条信息出现与并查集中记录的信息不符合时,则此信息不合法。

  由于a和b的值可能会非常大,所以我们还需要将它离散化。

代码

// poj1733
#include<algorithm>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<cstdio>
#include<cmath>
#define LL long long
#define inf 2147483640
#define Pi 3.1415926535898
#define free(a) freopen(a".in","r",stdin),freopen(a".out","w",stdout);
using namespace std;

int n,m,p[1000010],fa[1000010],s[1000010],x[1000010],y[1000010],pos[1000010],cnt;
char ch[10];

int binsearch(int x) {
	int tl=1,tr=cnt;
	while (tl<=tr) {
		int m=(tl+tr)>>1;
		if (x==pos[m]) return m;
		if (x>pos[m]) tl=m+1;
		else tr=m-1;
	}
}
int find(int x) {
	if (fa[x]==x) return x;
	int f=find(fa[x]);
	s[x]=(s[fa[x]]+s[x])&1;
	fa[x]=f;
	return fa[x];
}
void unnion(int i) {
	s[fa[x[i]]]=(s[x[i]]+p[i])&1;
	fa[fa[x[i]]]=y[i];
}
int main() {
	scanf("%d%d",&m,&n);
	for (int i=1;i<=n;i++) {
		scanf("%d%d",&x[i],&y[i]);
		scanf("%s",ch);
		if (ch[0]=='o') p[i]=1;
		x[i]--;
		pos[++cnt]=x[i];
		pos[++cnt]=y[i];
	}
	sort(pos+1,pos+1+cnt);
	pos[0]=1;
	for (int i=2;i<=cnt;i++) if (pos[i]!=pos[pos[0]]) {pos[0]++;pos[pos[0]]=pos[i];}
	cnt=pos[0];
	for (int i=1;i<=cnt;i++) fa[i]=i;
	for (int i=1;i<=n;i++) {
		x[i]=binsearch(x[i]);
		y[i]=binsearch(y[i]);
		if (find(x[i])==find(y[i])) {if (((s[x[i]]+s[y[i]])&1)!=p[i]) {printf("%d",i-1);return 0;}}
		else unnion(i);
		}
	printf("%d",n);
	return 0;
}

  

posted @ 2016-10-02 21:59  MashiroSky  阅读(383)  评论(0编辑  收藏  举报