IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 数据结构与算法 -> 图论3—差分约束 -> 正文阅读

[数据结构与算法]图论3—差分约束

差分约束

运用范围:(求解多个二元不等式组)

例如:

X 1 ? X 2 < = 0 X1-X2<=0 X1?X2<=0
X 3 ? X 4 < = 8 X3-X4<=8 X3?X4<=8
X 2 ? X 5 < = ? 9 X2-X5<=-9 X2?X5<=?9
X 3 ? X 1 < = 7 X3-X1<=7 X3?X1<=7
X 5 ? X 2 < = 0 X5-X2<=0 X5?X2<=0
. . . . . . ...... ......

原理:

  1. 对于这类不等式组,其解集要么无解,要么有无数解。(所有数同时加上X,其差值不会改变)
  2. 我们可以用最短路的思想解决这类问题

在对一张图跑完最短路算法后,对任意一条由u指向v的边都存在不等式:
Dis[v]<=Dis[u]+Len[u][v]
将这个不等式移项有:
Dis[v]-Dis[u]<=Len[u][v]
这与不等式: X 1 ? X 2 < = M X1-X2<=M X1?X2<=M神似。
所以我们只需要把 X 1 X1 X1当作v点, X 2 X2 X2当作u点,再向两点连一条边长为m的边即可。
最后这个不等式组就会变成一张图。

如何判断是否有解?

当你构造的这张图存在负权回路就表示无解
(若存在负权回路则Dis数组会无限减少,最终不满足不等式)

如何求出一组具体的解

  1. 新建一个原点 X 0 X0 X0,把 X 0 X0 X0向所有点连一条权值为0的边
  2. X 0 X0 X0为起点跑一遍最短路。
  3. 对每一个节点的Dis[i],就是 X i Xi Xi的一个可行解
  4. 若把Dis[ X 0 X0 X0]设置为A时,所有其它节点的值都会小于A(不等式定义)

例题

AC代码:

#include<stdio.h>
#include<bits/stdc++.h>
using namespace std;
#define H 500005
#define LL long long
int N,M;
int Dis[H],ED[H],LA[H],NT[H],LEN[H],tot,Cnt[H];
queue<int>Q;
void LB(int u,int v,int d){
	ED[++tot]=v;NT[tot]=LA[u];LA[u]=tot;LEN[tot]=d;
}
bool flag=1,MK[H];
void SPFA(int S){
	for(int i=1;i<=N;i++)Dis[i]=1000000000;
	Q.push(S);
	MK[S]=1;Cnt[S]++; 
	while(!Q.empty()){
		int x=Q.front();
		Q.pop();MK[x]=0;
		for(int i=LA[x];i;i=NT[i]){
			int y=ED[i];
			if(Dis[y]>Dis[x]+LEN[i]){
				Dis[y]=Dis[x]+LEN[i];
				if(!MK[y]){
					MK[y]=1;Cnt[y]++;
					if(Cnt[y]>N){
						flag=0;return;
					}
					Q.push(y);
				}
			}
		}
	}
}
int main(){
	scanf("%d%d",&N,&M);
	for(int i=1;i<=M;i++){
		int u,v,d;
		scanf("%d%d%d",&v,&u,&d);
		LB(u,v,d);
	}
	for(int i=1;i<=N;i++)LB(0,i,0);
	SPFA(0);
	if(!flag)puts("NO");
	else{
		for(int i=1;i<=N;i++)printf("%d ",Dis[i]);
	}
}
  数据结构与算法 最新文章
【力扣106】 从中序与后续遍历序列构造二叉
leetcode 322 零钱兑换
哈希的应用:海量数据处理
动态规划|最短Hamilton路径
华为机试_HJ41 称砝码【中等】【menset】【
【C与数据结构】——寒假提高每日练习Day1
基础算法——堆排序
2023王道数据结构线性表--单链表课后习题部
LeetCode 之 反转链表的一部分
【题解】lintcode必刷50题<有效的括号序列
上一篇文章      下一篇文章      查看所有文章
加:2022-02-16 13:22:22  更:2022-02-16 13:24:59 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/10 10:54:11-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码