洛谷P1073 Tarjan + 拓扑排序 // 构造分层图

https://www.luogu.org/problemnew/show/P1073

C国有 n n个大城市和 mm 条道路,每条道路连接这 nn个城市中的某两个城市。任意两个城市之间最多只有一条道路直接相连。这 mm 条道路中有一部分为单向通行的道路,一部分为双向通行的道路,双向通行的道路在统计条数时也计为 1 1条。C C国幅员辽阔,各地的资源分布情况各不相同,这就导致了同一种商品在不同城市的价格不一定相同。但是,同一种商品在同一个城市的买入价和卖出价始终是相同的。商人阿龙来到 CC 国旅游。当他得知同一种商品在不同城市的价格可能会不同这一信息之后,便决定在旅游的同时,利用商品在不同城市中的差价赚回一点旅费。设 CC 国 n 个城市的标号从 1~ n1 n,阿龙决定从 1 1号城市出发,并最终在 nn 号城市结束自己的旅行。在旅游的过程中,任何城市可以重复经过多次,但不要求经过所有 nn 个城市。阿龙通过这样的贸易方式赚取旅费:他会选择一个经过的城市买入他最喜欢的商品――水晶球,并在之后经过的另一个城市卖出这个水晶球,用赚取的差价当做旅费。由于阿龙主要是来 CC 国旅游,他决定这个贸易只进行最多一次,当然,在赚不到差价的情况下他就无需进行贸易。假设 C C国有 55个大城市,城市的编号和道路连接情况如下图,单向箭头表示这条道路为单向通行,双向箭头表示这条道路为双向通行。假设 1~n1 n 号城市的水晶球价格分别为 4,3,5,6,14,3,5,6,1。阿龙可以选择如下一条线路:11->22->33->55,并在 2 2号城市以 33 的价格买入水晶球,在 33号城市以 5 5的价格卖出水晶球,赚取的旅费数为 2。阿龙也可以选择如下一条线路 11->44->55->44->55,并在第1 1次到达 55 号城市时以 1 1的价格买入水晶球,在第 22 次到达 44 号城市时以 66 的价格卖出水晶球,赚取的旅费数为 55。现在给出 n n个城市的水晶球价格,mm 条道路的信息(每条道路所连接的两个城市的编号以及该条道路的通行情况)。请你告诉阿龙,他最多能赚取多少旅费。
题意

 

第一眼看觉得是先缩点然后DAG图上跑一边拓扑排序,除了敲起来手有点酸之外没有什么难度。就直接过了。

#include <map>
#include <set>
#include <ctime>
#include <cmath>
#include <queue>
#include <stack>
#include <vector>
#include <string>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <sstream>
#include <iostream>
#include <algorithm>
#include <functional>
using namespace std;
#define For(i, x, y) for(int i=x;i<=y;i++)  
#define _For(i, x, y) for(int i=x;i>=y;i--)
#define Mem(f, x) memset(f,x,sizeof(f))  
#define Sca(x) scanf("%d", &x)
#define Sca2(x,y) scanf("%d%d",&x,&y)
#define Sca3(x,y,z) scanf("%d%d%d",&x,&y,&z)
#define Scl(x) scanf("%lld",&x);  
#define Pri(x) printf("%d\n", x)
#define Prl(x) printf("%lld\n",x);  
#define CLR(u) for(int i=0;i<=N;i++)u[i].clear();
#define LL long long
#define ULL unsigned long long  
#define mp make_pair
#define PII pair<int,int>
#define PIL pair<int,long long>
#define PLL pair<long long,long long>
#define pb push_back
#define fi first
#define se second 
typedef vector<int> VI;
int read(){int x = 0,f = 1;char c = getchar();while (c<'0' || c>'9'){if (c == '-') f = -1;c = getchar();}
while (c >= '0'&&c <= '9'){x = x * 10 + c - '0';c = getchar();}return x*f;}
const double eps = 1e-9;
const int maxn = 1e5 + 10;
const int maxm = 5e5 + 10;
const int INF = 0x3f3f3f3f;
const int mod = 1e9 + 7; 
int N,M,K;
int val[maxn];
struct Edge{int to,next;
}edge[maxm * 2];
int head[maxn],tot;
void init(){for(int i = 0 ; i <= N + 1; i ++) head[i] = -1;tot = 0;
}
void add(int u,int v){edge[tot].to = v;edge[tot].next = head[u];head[u] = tot++;
}
int Low[maxn],dfn[maxn],Stack[maxn],Belong[maxn],num[maxn];
int Index,top,scc;
bool Instack[maxn];
void Tarjan(int u){int v;Low[u] = dfn[u] = ++Index;Stack[top++] = u;Instack[u] = true;for(int i = head[u]; ~i; i = edge[i].next){v = edge[i].to;if(!dfn[v]){Tarjan(v);if(Low[u] > Low[v]) Low[u] = Low[v];}else if(Instack[v] && Low[u] > dfn[v]){Low[u] = dfn[v];}}if(Low[u] == dfn[u]){scc++;do{v = Stack[--top];Instack[v] = false;Belong[v] = scc;num[scc]++;}while(v != u);}
}
int MAX[maxn],MIN[maxn],ind[maxn],ans[maxn];
vector<int>P[maxn];
int main(){Sca2(N,M); init();for(int i = 1; i <= N ; i ++) Sca(val[i]);for(int i = 1; i <= M ; i ++){int u,v,w; Sca3(u,v,w);add(u,v);if(w == 2) add(v,u);}for(int i = 1; i <= N ; i ++) if(!dfn[i]) Tarjan(i);for(int i = 1; i <= scc; i ++){MAX[i] = -INF; MIN[i] = INF;}for(int i = 1; i <= N ; i ++){int u = Belong[i];MAX[u] = max(MAX[u],val[i]);MIN[u] = min(MIN[u],val[i]);for(int j = head[i]; ~j; j = edge[j].next){int v = Belong[edge[j].to];if(u == v) continue;P[u].push_back(v);ind[v]++;    }}queue<int>Q;for(int i = 1; i <= scc; i ++) if(!ind[i]) Q.push(i);while(!Q.empty()){int u = Q.front(); Q.pop();ans[u] = max(ans[u],MAX[u] - MIN[u]);for(int i = 0; i < P[u].size(); i ++){int v = P[u][i];ind[v]--;MIN[v] = min(MIN[v],MIN[u]);ans[v] = max(ans[v],ans[u]);if(!ind[v]) Q.push(v);}}Pri(ans[Belong[N]]);return 0;
}
缩点 + 拓扑排序

 

后来发现还有一种更加新奇的解法,构造分层图,将原图变为一个带权图,权的意思是商人走这条路要得到的钱(负数代表花费),一开始的图所有边权都是0,表示这个商人什么也不买,走来走去不亏也不赚。然后我们对于每个点v,构造一条边通向v + N,边权为-val[v],表示商人在这个点买了一个水晶球,1 + N ~ N + N就是出现的第二个图,对于第二图,也像第一条路一样构造出相同的边权为0的边,表示商人买了水晶球之后依然可以不买也不卖的在这个图上走来走去,同理,我们再v + N ~ v + N + N构造一条边权为val[v]的边,表示商人买完之后在这个点又卖出去了,同第二个图一样,构造出第三个图,最后题意要求到达终点,所以可取的点只有N和3 * N,即第一个图和第三个图的终点,跑一边最短长路即可。

由于边权有正有负的,不能用Dijkstra跑,无向图也不可以拓扑排序dp,所以只能SPFA(大概是这个做法的唯一缺点)

#include <map>
#include <set>
#include <ctime>
#include <cmath>
#include <queue>
#include <stack>
#include <vector>
#include <string>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <sstream>
#include <iostream>
#include <algorithm>
#include <functional>
using namespace std;
#define For(i, x, y) for(int i=x;i<=y;i++)  
#define _For(i, x, y) for(int i=x;i>=y;i--)
#define Mem(f, x) memset(f,x,sizeof(f))  
#define Sca(x) scanf("%d", &x)
#define Sca2(x,y) scanf("%d%d",&x,&y)
#define Sca3(x,y,z) scanf("%d%d%d",&x,&y,&z)
#define Scl(x) scanf("%lld",&x);  
#define Pri(x) printf("%d\n", x)
#define Prl(x) printf("%lld\n",x);  
#define CLR(u) for(int i=0;i<=N;i++)u[i].clear();
#define LL long long
#define ULL unsigned long long  
#define mp make_pair
#define PII pair<int,int>
#define PIL pair<int,long long>
#define PLL pair<long long,long long>
#define pb push_back
#define fi first
#define se second 
typedef vector<int> VI;
int read(){int x = 0,f = 1;char c = getchar();while (c<'0' || c>'9'){if (c == '-') f = -1;c = getchar();}
while (c >= '0'&&c <= '9'){x = x * 10 + c - '0';c = getchar();}return x*f;}
const double eps = 1e-9;
const int maxn = 3e5 + 10;
const int maxm = 2e6 + 10;
const int INF = 0x3f3f3f3f;
const int mod = 1e9 + 7; 
int N,M,K;
int head[maxn],tot;
struct Edge{int to,next,dis;
}edge[maxm];
void init(){for(int i = 0 ; i <= 3 * N + 1; i ++) head[i] = -1;tot = 0;
}
void add(int u,int v,int w){edge[tot].to = v;edge[tot].next = head[u];edge[tot].dis = w;head[u] = tot++;
}
int val[maxn];
int dis[maxn],vis[maxn];
int SPFA(int s,int t){for(int i = 0 ; i <= 3 * N + 1; i ++) dis[i] = -INF;dis[1] = 0;queue<int>Q;Q.push(s);while(!Q.empty()){int u = Q.front(); Q.pop();vis[u] = 0;for(int i = head[u]; ~i ; i = edge[i].next){int v = edge[i].to;if(dis[v] < dis[u] + edge[i].dis){dis[v] = dis[u] + edge[i].dis;if(!vis[v]){vis[v] = 1;Q.push(v);}}} }if(dis[t] == -INF) dis[t] = 0;return dis[t];
}
int main(){Sca2(N,M); init();for(int i = 1; i <= N ; i ++){Sca(val[i]);add(i,i + N,-val[i]);add(i + N,i + N + N,val[i]);} for(int i = 1; i <= M ; i ++){int u,v,w; Sca3(u,v,w);add(u,v,0);add(u + N,v + N,0);add(u + N + N,v + N + N,0);if(w == 2){add(v,u,0);add(v + N,u + N,0);add(v + N,u + N + N,0);} }int t = 3 * N + 1,s = 1;add(N,t,0); add(3 * N,t,0);Pri(SPFA(s,t));return 0;
}

 

转载于:https://www.cnblogs.com/Hugh-Locke/p/10330356.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/264898.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

https网络编程——SSL的加密和解密过程

参考&#xff1a;SSL的加密和解密过程 地址&#xff1a;https://qingmu.blog.csdn.net/article/details/108214105?spm1001.2014.3001.5502 目录现在的加密/解密技术主要有三种&#xff1a;对称加密&#xff0c;非对称加密&#xff0c;和单向加密加密解密的流程图现在的加密/解…

IIS网站的权限设置问题

2019独角兽企业重金招聘Python工程师标准>>> IIS中的权限与NTFS权限设置的区别&#xff1a; 当浏览器访问被IIS禁止的页面时&#xff0c;返回404错误页面 但浏览器访问被NTFS禁止的文件时&#xff0c;提示用户登录界面 转载于:https://my.oschina.net/changeme/blog…

关于.Net2.0下配置架构的使用

上次用到配置文件,就花了一些时间研究了一下.Net2.0下的配置文件架构,当时感觉确实很强大,完善,但看的有些头晕.迷迷糊糊把实现了要求,就没有再深入研究.最近,想在配置文件里实现一个复杂的配置,多层次嵌套的配置文件,再把.Net中的配置文档研究了一下&#xff0c;经过这两次的研…

煲仔

湖南的煲仔似乎在我毕业工作的那年侵占了大街小巷&#xff0c;满街的煲仔店向外热滋滋的喷着香味&#xff0c;便宜又好味。那几年在外的觅食&#xff0c;除了常德津市牛肉米饭外&#xff0c;煲仔似乎是我唯一的选择。将米放入煲内&#xff0c;加水在火上煮至半熟&#xff0c;再…

https网络编程——中继(负载均衡)工作原理

参考&#xff1a;中继&#xff08;负载均衡&#xff09;工作原理 地址&#xff1a;https://qingmu.blog.csdn.net/article/details/108217055?spm1001.2014.3001.5502 中继&#xff08;负载均衡&#xff09;工作原理 在我们当前社会&#xff0c;比如我们上某宝&#xff0c;某…

全套支付宝系统架构(内部架构图)【收藏】

据说这是对支付宝系统体系最全最强解析&#xff0c;推荐收藏学习&#xff01; 转载于:https://www.cnblogs.com/SH-xuliang/p/10340745.html

iOS学习之iOS沙盒(sandbox)机制和文件操作之NSFileManager

2019独角兽企业重金招聘Python工程师标准>>> 1、在Documents里创建目录 创建一个叫test的目录,先找到Documents的目录&#xff0c; [cpp] view plain copy NSArray *paths NSSearchPathForDirectoriesInDomains(NSDocumentDirectory, NSUserDomainMask, YES); …

https网络编程——使用openssl库自建根证书

参考&#xff1a;如何自建根证书&#xff1f;使用openssl库自建根证书带图详解 地址&#xff1a;https://qingmu.blog.csdn.net/article/details/108217572?spm1001.2014.3001.5502 目录根证书的普通用途自建根证书步骤1、创建一个目录&#xff0c;存放所有证书有关资料2、进入…

angular接口传参

1、service文件 创建xxx.service.ts文件 import { Injectable, Inject } from angular/core;import { Observable } from rxjs;import { map } from rxjs/operators;import { HttpClient } from angular/common/http;Injectable({ providedIn: root})export class ErrorCond…

https网络编程——如何建立利用根证书(凭证)签发建立中继证书(凭证)详解

参考&#xff1a;如何建立利用根证书&#xff08;凭证&#xff09;签发建立中继证书&#xff08;凭证&#xff09;详解 地址&#xff1a;https://qingmu.blog.csdn.net/article/details/108221568?spm1001.2014.3001.5502 目录在建立中继之前需要自建根证书建立根证书的具体步…

NURBS曲线与曲面

B样条方法在表示与设计自由型曲线曲面形状时显示了强大的威力&#xff0c;然而在表示与设计初等曲线曲面时时却遇到了麻烦。因为B样条曲线包括其特例的Bezier曲线都不能精确表示出抛物线外的二次曲线&#xff0c;B样条曲面包括其特例的Bezier曲面都不能精确表示出抛物面外的二次…

https网络编程——如何利用中继证书(凭证)建立服务器证书

参考&#xff1a;如何利用中继证书&#xff08;凭证&#xff09;建立服务器证书 地址&#xff1a;https://qingmu.blog.csdn.net/article/details/108225569?spm1001.2014.3001.5502 目录建立服务器证书的前提是要建立中继证书建立服务器证书的具体步骤1、建立一个目录&#x…

上传图片

2019独角兽企业重金招聘Python工程师标准>>> private File imageFile;// 上传文件名称private String imageFileFileName;// 上传文件类型private String imageFileContextType; InputStream is new FileInputStream(imageFile);String suffixName imageFileFileN…

https网络编程——如何利用中继证书(凭证)建立客户端证书

参考&#xff1a;如何利用中继证书&#xff08;凭证&#xff09;建立客户端证书 地址&#xff1a;https://qingmu.blog.csdn.net/article/details/108226592?spm1001.2014.3001.5502 目录建立客户端证书的前提是要建立中继证书建立客户端证书的具体步骤1、建立一个目录&#x…

2019.2.4 nfs原理和安装实验

NFS 访问一个本地文件还是NFS共享文件对于客户端而言都是透明的&#xff0c;当文件打开的瞬间&#xff0c;内核会作出一个决定&#xff0c;如果是本地文件内核会将本地NFS共享文件内核会将NFS共享文件的所有引用传递给——》NFS客户端枢中 NFS客户端是通过TCP/IP协议及模块向NF…

形容人的内核是什么意思_成语雪泥鸿爪是形容什么的?雪泥鸿爪什么意思?蚂蚁庄园2020年12月10日答案...

斑马线和斑马什么关系&#xff1f;大家都知道斑马和斑马线&#xff0c;但是两者之间有什么关系&#xff1f;蚂蚁庄园12月10日提到了这个问题&#xff0c;我们一起来看看正确答案吧。问题&#xff1a;斑马线和斑马有什么关系&#xff1f;答案&#xff1a;横线酷似斑马纹答案解析…

.Net 2.0里有一个有用的新功能:迭代器

下面内容节选至MSDN2005。迭代器&#xff08;C# 编程指南&#xff09; 迭代器是 C# 2.0 中的新功能。迭代器是方法、get 访问器或运算符&#xff0c;它使您能够在类或结构中支持 foreach 迭代&#xff0c;而不必实现整个 IEnumerable 接口。您只需提供一个迭代器&#xff0c;即…

MySQL 服务器变量 数据操作DML-视图

原文:MySQL 服务器变量 数据操作DML-视图SQL语言的组成部分 常见分类&#xff1a; DDL&#xff1a;数据定义语言 DCL&#xff1a;数据控制语言&#xff0c;如授权 DML&#xff1a;数据操作语言 其它分类&#xff1a; 完整性定义语言&#xff1a;DDL的一部分功能约束约束&#x…

kafka内存不断增加_为什么 Kafka 能这么快的 6 个原因

无论 kafka 作为 MQ 也好&#xff0c;作为存储层也罢&#xff0c;无非就是两个功能(好简单的样子)&#xff0c;一是 Producer 生产的数据存到 broker&#xff0c;二是 Consumer 从 broker 读取数据。那 Kafka 的快也就体现在读写两个方面了&#xff0c;下面我们就聊聊 Kafka 快…

https网络编程——DNS域名解析获取IP地址

参考&#xff1a;DNS域名解析 地址&#xff1a;https://qingmu.blog.csdn.net/article/details/115825036?spm1001.2014.3001.5502 1、原理 我在在通过域名解析获取IP的过程中一般使用的是DNS域名解析。 DNS协议是一种应用层协议&#xff0c;他是基于UDP来实现的。 2、代码…