題解 P4782 【【模板】2-SAT 問題】

nn 個布爾變量 x1x_1 xn\sim x_n ,另有 mm 個需要滿足的條件,每個條件的形式都是 「xix_itrue / falsexjx_j 爲 true / false」。比如 「x1x_1 爲真或 x3x_3 爲假」、「x7x_7 爲假或 x2x_2 爲假」。

2-SAT 問題的目標是給每個變量賦值使得所有條件得到滿足。

前置知識:

代碼:

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
template<typename T>inline void read(T &FF){
	T RR=1;FF=0;char CH=getchar();
	for(;!isdigit(CH);CH=getchar())if(CH=='-')RR=-1;
	for(;isdigit(CH);CH=getchar())FF=(FF<<1)+(FF<<3)+(CH^48);
	FF*=RR;
}
const int MAXN=3e6+10;
int n,m,a,b,a1,b1,dfn[MAXN],low[MAXN],tot,s[MAXN],sp,sccnum[MAXN],scccnt;
vector<int>E[MAXN];
void tarjan(int u){
	s[sp++]=u;
	dfn[u]=low[u]=++tot;
	for(auto v:E[u])
		if(!dfn[v]){
			tarjan(v);
			low[u]=min(low[u],low[v]);
		}else if(!sccnum[v]){
			low[u]=min(low[u],dfn[v]);
		}
	if(dfn[u]==low[u]){
		scccnt++;
		do{
			sccnum[s[--sp]]=scccnt;
		}while(s[sp]!=u);
	}
}
int main(){
	read(n);read(m);
	for(int i=1;i<=m;i++){
		read(a);read(a1);read(b);read(b1);
        E[a+(a1^1)*n].push_back(b+b1*n);
        E[b+(b1^1)*n].push_back(a+a1*n);
	}
	for(int i=1;i<=(n<<1);i++)
		if(!dfn[i])tarjan(i);
	for(int i=1;i<=(n<<1);i++)
		if(sccnum[i]==sccnum[i+n])return puts("IMPOSSIBLE"),0;
	puts("POSSIBLE");
	for(int i=1;i<=n;i++)cout<<(sccnum[i]>sccnum[i+n])<<" ";
	return 0;
}
/*
3 1
1 1 3 0
*/
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章