首页 > 其他分享 >P6378 [PA2010] Riddle-2sat优化建图

P6378 [PA2010] Riddle-2sat优化建图

时间:2023-10-20 18:00:44浏览次数:29  
标签:Riddle P6378 pre int neg rep PA2010 low addEdge

P6378 [PA2010] Riddle-2sat优化建图

\(n\) 个点 \(m\) 条边的无向图被分成 \(k\) 个部分。每个部分包含一些点。

请选择一些关键点,使得每个部分恰有一个关键点,且每条边至少有一个端点是关键点。

\(1\leq n,m\leq 10^6\)


边的限制

用 \(n\) 个变量 \(x_1\dots x_n\) ,其中 \(x_i\) 表示命题:“ \(i\) 是关键点”,与此同时也需要另外 \(n\) 个变量 \(\neg x_1,\dots,\neg x_n\) ,自然地 \(\neg x_i\) 表示命题”\(i\) 不是关键点“。

那么对于一条边的约束,就变成形如:

\[x_i\lor x_j=(\neg x_i\to x_j)\land (\neg x_j\to x_i) \]

的约束,把 \(x_1\dots,x_n,\neg x_1,\dots,\neg x_n\) 看成 \(2n\) 个结点,问题约束就变成了要选择一些点,使得如果选了点 \(x\) ,那么所有 \(x\) 能到达的点也要选,以及很明显不能同时选择 \(x_i\) 和 \(\neg x_i(\forall i=1,\dots,n)\)。

每个部分的限制

每个部分 \(S\) 中的元素至多选择一个,即如果选择了 \(x\) ,那么要向所有\(\neg y(y\in S,y\neq x)\) 连边,暴力连边是 \(O(n^2)\) 的显然不行。

但这种建图其实也很常见,每次给集合内除了自己之外的点连边,假设 \(S\) 中的点有序,排成 \(a_1,\dots,a_w\) ,那么对于每个 \(a_i\) 只要给前 \(i-1\) 个点和 \(i+1\) 往后的 \(n-i\) 个点连边,所以可以额外创建 \(w\) 个点表示前缀,\(w\) 个点表示后缀。

建图代码:

rep(tc,1,k){
    int w,c;cin>>w;
    rep(i,1,w)cin>>a[i];
    rep(i,1,w)pre[i]=++tot;
    rep(i,1,w)suf[i]=++tot;
    rep(i,1,w){
        addEdge(pre[i],a[i]+n);
        addEdge(suf[i],a[i]+n);
        if(i>1){
            addEdge(pre[i],pre[i-1]);
            addEdge(a[i],pre[i-1]);
        }
        if(i<w){
            addEdge(suf[i],suf[i+1]);
            addEdge(a[i],suf[i+1]);
        }
    }
}

完整代码:

#include<bits/stdc++.h>
#define rep(i,a,b) for(int i=(a);i<=(b);i++)
#define fastio ios_base::sync_with_stdio(false);cin.tie(0);cout.tie(0)
using namespace std;
const int N=4e6+5;
vector<vector<int>> G,H;
int n,m,k,tot,pre[N],suf[N],a[N];
int bl[N],scc,dfscnt,dfn[N],low[N];
bool in_stack[N];
stack<int> S;
void tarjan(int x){
	low[x]=dfn[x]=++dfscnt;
	S.push(x);in_stack[x]=1;
	for(auto to:G[x]){
		if(!dfn[to]){
			tarjan(to);
			low[x]=min(low[x],low[to]);
		}else if(in_stack[to])
			low[x]=min(low[x],dfn[to]);
	}
	if(low[x]==dfn[x]){
		++scc;
		bool tag=0;
		while(1){
			bl[S.top()]=scc;
			if(S.top()==x)tag=1;
			in_stack[S.top()]=0;
			S.pop();
			if(tag)break;
		}
	}
}
void addEdge(int x,int y){G[x].push_back(y);}
int main(){
	fastio;
	cin>>n>>m>>k;
	G=vector<vector<int>>(4*n+5);
	rep(i,1,m){
		int x,y;cin>>x>>y;
		addEdge(x+n,y);
		addEdge(y+n,x);
	}
	tot=2*n;
	rep(tc,1,k){
		int w,c;cin>>w;
		rep(i,1,w)cin>>a[i];
		rep(i,1,w)pre[i]=++tot;
		rep(i,1,w)suf[i]=++tot;
		rep(i,1,w){
			addEdge(pre[i],a[i]+n);
			addEdge(suf[i],a[i]+n);
			if(i>1){
				addEdge(pre[i],pre[i-1]);
				addEdge(a[i],pre[i-1]);
			}
			if(i<w){
				addEdge(suf[i],suf[i+1]);
				addEdge(a[i],suf[i+1]);
			}
		}
	}
	rep(i,1,tot)if(!bl[i])tarjan(i);
	bool bad=0;
	rep(i,1,n)if(bl[i]==bl[i+n])bad=1;
	if(bad)cout<<"NIE";
	else cout<<"TAK";
	return 0;
}

标签:Riddle,P6378,pre,int,neg,rep,PA2010,low,addEdge
From: https://www.cnblogs.com/yoshinow2001/p/17777689.html

相关文章

  • CF1466I The Riddle of the Sphinx
    基本思路明示了在二进制下考虑问题,我们大体的思路就是从高往低依次确定最大的数二进制下每一位上的值。以下所述的「前缀」均指一个二进制数从高位到低位的一部分,一个元素的「前\(k\)位」表示二进制从高位到低位的前\(k\)位,\(res\)表示当前记录的最大前缀的长度。先看看操......