【模板】2-SAT

时间:2023-12-05 18:59:26

题目大意:给定 N 个点的 M 条约束,约束形式为:\(a_i \lor a_j = 1\)。

题解:拆点什么的就不说了,在求出一组解的时候,考虑到 Tarjan 找环的过程中,scc 染色是按照拓扑序的逆序来进行的,即:拓扑排序中最后被删除的节点的 cor 值最小。根据这个性质,在一定有解的情况下,对于任意一个 \(a_i\),应该取拓扑序大的值(Tarjan 染色小的值)作为最终结果,因为缩点之后的 DAG 上的边依然是选了前一个scc 就必然选与之相连的 scc。

代码如下

#include <bits/stdc++.h>
#define pb push_back
#define mp make_pair
#define all(x) x.begin(),x.end()
using namespace std;
typedef long long ll;
typedef pair<int,int> P;
const int inf=0x3f3f3f3f;
const int maxn=2e6+10; inline int read(){
int x=0,f=1;char ch;
do{ch=getchar();if(ch=='-')f=-1;}while(!isdigit(ch));
do{x=x*10+ch-'0';ch=getchar();}while(isdigit(ch));
return f*x;
} int n,m;
vector<int> G[maxn];
int dfs_clk,dfn[maxn],low[maxn],stk[maxn],top,in[maxn];
int scc,cor[maxn]; void tarjan(int u){
dfn[u]=low[u]=++dfs_clk;
stk[++top]=u,in[u]=1;
for(int i=0;i<G[u].size();i++){
int v=G[u][i];
if(!dfn[v])tarjan(v),low[u]=min(low[u],low[v]);
else if(in[v])low[u]=min(dfn[v],low[u]);
}
if(dfn[u]==low[u]){
++scc;int v;
do{
v=stk[top--],in[v]=0;
cor[v]=scc;
}while(u!=v);
}
} void read_and_parse(){
n=read(),m=read();
while(m--){
int i=read(),a=read(),j=read(),b=read();
if(a&&b)G[i].pb(j+n),G[j].pb(i+n);
else if(a&&!b)G[i].pb(j),G[j+n].pb(i+n);
else if(!a&&b)G[i+n].pb(j+n),G[j].pb(i);
else G[i+n].pb(j),G[j+n].pb(i);
}
}
void solve(){
for(int i=1;i<=n<<1;i++)if(!dfn[i])tarjan(i);
for(int i=1;i<=n;i++)if(cor[i]==cor[i+n])return (void)puts("IMPOSSIBLE");
puts("POSSIBLE");
for(int i=1;i<=n;i++)printf("%d ",!(cor[i]<cor[i+n]));
puts("");
}
int main(){
read_and_parse();
solve();
return 0;
}