Luogu P5960 【模板】差分约束算法

2021-04-21 03:29

阅读:426

标签:eof   pac   oge   反转   void   scanf   ons   set   lin   

gate

差分约束系统用来解决:
给出\(n\)个变量\(x_1...x_n\)\(m\)个形如\(x_i-x_j \le k\)\(k\)为任意常量)的式子,
\(x_1...x_n\)的一组可行解。

将式子变形为\(x_i\le x_j + k\),发现刚好符合三角形不等式\(dis[v]\le dis[u]+w[i]\)的形式。
对于每个式子,连边\(x_j \rightarrow x_i\),权值为\(k\)\(0\)号节点向每个节点连边,权值为\(0\)
转化为求单源最短路,\(dis[1]...dis[n]\)即为解。
由于可能出现负权边,用\(SPFA\)求解。若出现负环,则无解。

对于一些其他形式的式子,需要稍作转化:
\(x_i-x_j \ge k\) 将两边同时乘\(-1\),将不等号反转。
\(x_i-x_j = k\) 拆成两个式子\(x_i-x_j \le k\)\(x_i-x_j \ge k\)

\(code\)

#include
#include
#include
#include
#include
#define MogeKo qwq
using namespace std;

const int maxn = 1e5+10;
const int INF = 0x3f3f3f3f;

int n,m,x,y,z,cnt;
int head[maxn],to[maxn],nxt[maxn],w[maxn];
int dis[maxn],num[maxn];
bool vis[maxn];

void add(int x,int y,int z) {
	to[++cnt] = y;
	nxt[cnt] = head[x];
	head[x] = cnt;
	w[cnt] = z;
}

bool SPFA(int s) {
	memset(dis,INF,sizeof(dis));
	queue  q;
	dis[s] = 0;
	vis[s] = true;
	q.push(s);
	while(!q.empty()) {
		int u = q.front();
		q.pop();
		vis[u] = false;
		for(int i = head[u]; i; i = nxt[i]) {
			int v = to[i];
			if(dis[v] 

Luogu P5960 【模板】差分约束算法

标签:eof   pac   oge   反转   void   scanf   ons   set   lin   

原文地址:https://www.cnblogs.com/mogeko/p/13283007.html


评论


亲,登录后才可以留言!