题意:$2n$个位置排成一列,有一些位置已经填了数字($0\cdots n-1$中每个数字出现$0$次或$2$次),问是否存在一种填数方案使得用$n$个不相交的半圆可以把相同的数字连起来

首先把所有已经填了的数字的半圆画出来,如果两个半圆相交那么它们必须在异侧,在相交的半圆之间连边,如果不是二分图那么就无解

我们用$-1$代表未填数的位置,给$-1$的位置分配$1$代表这个位置向上连线,$0$代表这个位置向下连线

朴素的想法是:对于每个半圆$[l,r]$,枚举它在上方还是在下方,如果在上方,那么$[l,r]$中必须有偶数个$1$,如果在下方,那么$[l,r]$中必须有偶数个$0$(用$-1$的个数减去$0$的个数即可得到$1$的个数),并且因为对于每个不是$-1$的位置$i$,因为不能给它分配$1$,所以$[i,i]$中必须有偶数个$0$,最后,显然所有位置上必须有偶数个$1$

现在问题变为:给定一些区间$[l,r]$和对其中$1$的个数的奇偶性要求,问是否能满足,求异或前缀和后就变成对一些变量的异或值限制,直接dfs一遍看是否冲突即可

但是这样会很慢,考虑优化

如果一个半圆$[l,r]$不与其他半圆相交,当区间长度为偶数时,它放在上或下都要求区间中$1$的个数是偶数,当区间长度为奇数时,放在上或下可以使得区间中$1$的个数是奇数或是偶数,又因为它和其他半圆互不影响,所以我们这样处理:如果区间长度为奇数,不管它,如果区间长度为偶数,那么我们不用枚举它放在上还是下,直接令区间中$1$的个数是偶数即可

所以我们只用对二分图中大小$\geq2$的连通块枚举它的两半分别在上还是下,因为最多有$\frac n2$个半圆,所以最多有$\frac n4$个大小$\geq2$的连通块,所以总时间复杂度为$O(n2^{\frac n4})$

#include<stdio.h>
#include<string.h>
#include<algorithm>
#include<vector>
#include<string>
using namespace std;
struct seg{
	int l,r;
	seg(int a=0,int b=0){l=a;r=b;}
}e[30];
bool ints(seg a,seg b){
	if(a.l>b.l)swap(a,b);
	return b.l<a.r&&b.r>a.r;
}
vector<int>vt[30][2];
vector<int>sg;
int cnt;
struct graph{
	int h[30],nex[610],to[610],M,n;
	void reset(){
		M=0;
		memset(h,0,sizeof(h));
	}
	void ins(int a,int b){
		M++;
		to[M]=b;
		nex[M]=h[a];
		h[a]=M;
	}
	void add(int a,int b){
		ins(a,b);
		ins(b,a);
	}
	int c[30];
	bool dfs(int x,int f){
		if(~c[x])return c[x]==f;
		vt[cnt][f].push_back(x);
		c[x]=f;
		for(int i=h[x];i;i=nex[i]){
			if(!dfs(to[i],f^1))return 0;
		}
		return 1;
	}
	bool gao(int n){
		sg.clear();
		memset(c,-1,sizeof(c));
		cnt=0;
		for(int i=1;i<=n;i++){
			if(c[i]==-1){
				if(h[i]){
					vt[cnt][0].clear();
					vt[cnt][1].clear();
					if(!dfs(i,0))return 0;
					cnt++;
				}else
					sg.push_back(i);
			}
		}
		return 1;
	}
}g;
struct graph2{
	int h[60],nex[210],to[210],v[210],M;
	void reset(){
		M=0;
		memset(h,0,sizeof(h));
	}
	void ins(int a,int b,int c){
		M++;
		to[M]=b;
		v[M]=c;
		nex[M]=h[a];
		h[a]=M;
	}
	void add(int a,int b,int c){
		ins(a,b,c);
		ins(b,a,c);
	}
	int c[60];
	bool dfs(int x,int f){
		if(~c[x])return c[x]==f;
		c[x]=f;
		for(int i=h[x];i;i=nex[i]){
			if(!dfs(to[i],f^v[i]))return 0;
		}
		return 1;
	}
	bool gao(int n){
		memset(c,-1,sizeof(c));
		for(int i=0;i<=n;i++){
			if(c[i]==-1&&!dfs(i,0))return 0;
		}
		return 1;
	}
}g2;
int a[60],s[60];
class DisjointSemicircles{
	public:
		string getPossibility(vector<int>v){
			int n,i,j,M;
			n=v.size();
			M=0;
			for(i=1;i<=n;i++){
				a[i]=v[i-1];
				s[i]=s[i-1]+(a[i]==-1);
				if(~a[i]){
					for(j=1;j<i;j++){
						if(a[j]==a[i])break;
					}
					if(i!=j)e[++M]=seg(j,i);
				}
			}
			g.reset();
			for(i=1;i<M;i++){
				for(j=i+1;j<=M;j++){
					if(ints(e[i],e[j]))g.add(i,j);
				}
			}
			if(!g.gao(M))return"IMPOSSIBLE";
			for(i=0;i<1<<cnt;i++){
				g2.reset();
				for(int x:sg){
					if((e[x].r-e[x].l)&1)g2.add(e[x].l-1,e[x].r,0);
				}
				for(j=0;j<cnt;j++){
					for(int x:vt[j][i>>j&1])g2.add(e[x].l-1,e[x].r,0);
					for(int x:vt[j][~i>>j&1])g2.add(e[x].l-1,e[x].r,(s[e[x].r]-s[e[x].l-1])&1);
				}
				for(j=1;j<=n;j++){
					if(~a[j])g2.add(j-1,j,0);
				}
				g2.add(0,n,0);
				if(g2.gao(n))return"POSSIBLE";
			}
			return"IMPOSSIBLE";
		}
};
/*
int main(){
	int x;
	DisjointSemicircles cl;
	vector<int>v;
	for(scanf("%d",&x);x!=-2;scanf("%d",&x))v.push_back(x);
	puts(cl.getPossibility(v).c_str());
}
*/

[SRM568]DisjointSemicircles的更多相关文章

  1. Topcoder口胡记 SRM 562 Div 1 ~ SRM 599 Div 1

    据说做TC题有助于提高知识水平? :) 传送门:https://284914869.github.io/AEoj/index.html 转载请注明链接:http://www.cnblogs.com/B ...

  2. TC做题笔记

    SRM593 Div1Medium--May The Best Pet Win(bitset优化) Description 给出n个元素取值的max.min,把这n个元素分割成两个集合,求如何分割使两 ...

随机推荐

  1. 关于Linux下s、t、i、a权限

    文件权限除了r.w.x外还有s.t.i.a权限: 首先我们利用umask查看系统的权限为四位,首位就是特殊权限位,SetUID为4,SetGID为2,t为1 [root@iz2ze46xi6pjjj6 ...

  2. upupw注入by pass

    http:' and updatexml(null,concat(0x5c,(/*!00000select SCHEMA_name*/from/*!information_schema*/.schem ...

  3. 去掉每行的特定字符py脚本

    百度下载一个脚本的时候遇到那么一个情况.每行的开头多了一个空格.https://www.0dayhack.com/post-104.html 一个个删就不要说了,很烦.于是就有了下面这个脚本. #! ...

  4. imx6设备树pinctrl解析【转】

    转自:http://blog.csdn.net/michaelcao1980/article/details/50730421 版权声明:本文为博主原创文章,未经博主允许不得转载. 最近在移植linu ...

  5. EF添加ADO.NET实体模型处直接选择Oracle数据源

    上一文介绍了如何下载Mysql for vs Tools来进行Mysql的ADO.NET实体模型数据源选择,今天将Oracle的测试了下.步骤如下: 1.在你项目Model层中nuget安装选中项 2 ...

  6. 计数排序的实现--适用于元素均较小的seq

    今天无聊就打算把所有的排序算法都看一遍... 计数排序的时间复杂度是O(n),在算法导论中,用决策树模型中论证了,比较排序的情况为nlogn的复杂度.而计数排序的时间复杂度小于他的原因就是它不需要进行 ...

  7. UVA 11076 Add Again

    题目链接:UVA-33478 题意为给定n个数,求这n个数能组成的所有不同的排列组成的数字的和. 思路:发现对于任意一个数字,其在每一位出现的次数是相同的.换言之,所有数字的每一位相加的和是相同的. ...

  8. 2017多校第9场 HDU 6166 Senior Pan 堆优化Dij

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6166 题意:给你一个有向图,然后给你k个点,求其中一个点到另一个点的距离的最小值. 解法:枚举二进制位 ...

  9. swift 动态获取类, 获取命名空间

    在做swift开发中很多时候会动态加载控制器的类, 可以让app更加灵活显示界面信息 一般情况下都是服务器返回显示的控制器类name然后动态显示, 但是服务器返回的类name是string, 怎么转换 ...

  10. (转)关于bootstrap, boosting, bagging,Rand forest

    转自:https://blog.csdn.net/jlei_apple/article/details/8168856 这两天在看关于boosting算法时,看到一篇不错的文章讲bootstrap, ...