题目链接

先\(Tarjan\)缩点,记录每个环内的最大值和最小值。

然后跑拓扑排序,\(Min[u]\)表示到\(u\)的最小值,\(ans[u]\)表示到\(u\)的答案,\(Min\)和\(ans\)都在拓扑排序中更新和传递。

最终答案就是\(ans[n]\)。

\(100\)多行敲着心累

#include <cstdio>
#include <cstring>
#define Open(s) freopen(s".in","r",stdin);freopen(s".out","w",stdout);
#define Close fclose(stdin);fclose(stdout);
namespace IO{
int xjc; char ch;
inline int read(){
xjc = 0; ch = getchar();
while(ch < '0' || ch > '9') ch = getchar();
while(ch >= '0' && ch <= '9'){ xjc = xjc * 10 + ch - '0'; ch = getchar(); }
return xjc;
}
}using namespace IO;
inline int max(int a, int b){
return a > b ? a : b;
}
inline int min(int a, int b){
return a > b ? b : a;
}
const int MAXN = 100010;
const int MAXM = 500010;
struct Queue{
int s[MAXN];
int head, tail;
inline void push(int x){
s[++tail] = x;
}
inline int pop(){
return s[++head];
}
inline int size(){
return tail - head;
}
}q;
struct Edge{
int from, next, to;
};
struct Graph{
int head[MAXN], num;
Edge e[MAXM << 1];
inline void Add(int from, int to){
e[++num].to = to; e[num].from = from; e[num].next = head[from]; head[from] = num;
}
}G, T, P;
int dfn[MAXN], low[MAXN], id, vis[MAXN], stack[MAXN], w[MAXN], top, cnt, belong[MAXN], v[MAXN];
int minw[MAXN], maxw[MAXN], now, in[MAXN], Min[MAXN], ans[MAXN];
void Tarjan(int u){
dfn[u] = low[u] = ++id; vis[u] = 1; stack[++top] = u;
for(int i = G.head[u]; i; i = G.e[i].next)
if(!dfn[G.e[i].to]){
Tarjan(G.e[i].to);
low[u] = min(low[u], low[G.e[i].to]);
}
else if(vis[G.e[i].to])
low[u] = min(low[u], dfn[G.e[i].to]);
if(dfn[u] == low[u]){
++cnt;
do{
now = stack[top--];
vis[now] = 0;
belong[now] = cnt;
minw[cnt] = min(minw[cnt], w[now]);
maxw[cnt] = max(maxw[cnt], w[now]);
P.Add(cnt, now);
}while(now != u);
}
}
int n, m, a, b, c;
int main(){
Open("trade");
memset(minw, 127, sizeof minw);
memset(Min, 127, sizeof Min);
n = read(); m = read();
for(int i = 1; i <= n; ++i)
w[i] = read();
for(int i = 1; i <= m; ++i){
a = read(); b = read(); c = read();
G.Add(a, b);
if(c == 2) G.Add(b, a);
}
for(int i = 1; i <= n; ++i)
if(!dfn[i])
Tarjan(i);
for(int i = 1; i <= cnt; ++i){
for(int j = P.head[i]; j; j = P.e[j].next){
int u = P.e[j].to;
for(int k = G.head[u]; k; k = G.e[k].next)
if(belong[G.e[k].to] != i && !v[belong[G.e[k].to]]){
v[belong[G.e[k].to]] = 1;
T.Add(i, belong[G.e[k].to]);
++in[belong[G.e[k].to]];
}
}
for(int j = P.head[i]; j; j = P.e[j].next){
int u = P.e[j].to;
for(int k = G.head[u]; k; k = G.e[k].next)
v[belong[G.e[k].to]] = 0;
}
}
q.push(belong[1]);
while(q.size()){
now = q.pop();
Min[now] = min(Min[now], minw[now]);
ans[now] = max(ans[now], maxw[now] - Min[now]);
for(int i = T.head[now]; i; i = T.e[i].next){
int v = T.e[i].to;
if(!(--in[v])) q.push(v);
Min[v] = min(Min[v], Min[now]);
ans[v] = max(ans[v], ans[now]);
}
}
printf("%d\n", ans[belong[n]]);
return 0;
}

【洛谷 P1073】 最优贸易 (Tarjan缩点+拓扑排序)的更多相关文章

  1. 洛谷 P1073 最优贸易 解题报告

    P1073 最优贸易 题目描述 \(C\)国有\(n\)个大城市和\(m\)条道路,每条道路连接这\(n\)个城市中的某两个城市.任意两个城市之间最多只有一条道路直接相连.这\(m\)条道路中有一部分 ...

  2. 洛谷P1073 最优贸易==codevs1173 最优贸易

    P1073 最优贸易 题目描述 C 国有 n 个大城市和 m 条道路,每条道路连接这 n 个城市中的某两个城市.任意两个 城市之间最多只有一条道路直接相连.这 m 条道路中有一部分为单向通行的道路,一 ...

  3. 洛谷——P1073 最优贸易

    P1073 最优贸易 n 个城市间以 m 条有向道路连接, 小 T 从 1 号城市出发, 将要去往 n 号城市.小 T 观察到一款商品 Z 在不同的城市的价格可能不尽相同,小 T 想要在旅行中的某一个 ...

  4. 洛谷 P1073 最优贸易 最短路+SPFA算法

    目录 题面 题目链接 题目描述 输入输出格式 输入格式 输出格式 输入输出样例 输入样例 输出样例 说明 思路 AC代码 题面 题目链接 P1073 最优贸易 题目描述 C国有 $ n $ 个大城市和 ...

  5. 洛谷P1073 最优贸易 [图论,DP]

    题目传送门 最优贸易 题目描述 C 国有n 个大城市和m 条道路,每条道路连接这n 个城市中的某两个城市.任意两个城市之间最多只有一条道路直接相连.这m 条道路中有一部分为单向通行的道路,一部分为双向 ...

  6. 洛谷 P1073 最优贸易

    题目描述 CC C 国有 n n n 个大城市和 m mm 条道路,每条道路连接这 nnn 个城市中的某两个城市.任意两个城市之间最多只有一条道路直接相连.这 mmm 条道路中有一部分为单向通行的道路 ...

  7. 洛谷 P1073 最优贸易 & [NOIP2009提高组](反向最短路)

    传送门 解题思路 很长的题,实际上在一个有向图(点有点权)中求一个从起点1到终点n的路径,使得这条路径上点权最大的点与点权最小的点的差值最大(要求必须从点权较小的点能够走到点权较大的点). ——最短路 ...

  8. NOIP2009 codevs1173 洛谷P1073 最优贸易

    Description: 国有 n 个大城市和 m 条道路,每条道路连接这 n 个城市中的某两个城市.任意两个城市之间最多只有一条道路直接相连.这 m 条道路中有一部分为单向通行的道路,一部分为双向通 ...

  9. [NOIP2009] 提高组 洛谷P1073 最优贸易

    题目描述 C 国有 n 个大城市和 m 条道路,每条道路连接这 n 个城市中的某两个城市.任意两个 城市之间最多只有一条道路直接相连.这 m 条道路中有一部分为单向通行的道路,一部分 为双向通行的道路 ...

  10. 洛谷P1073最优贸易——双向取值

    题目:https://www.luogu.org/problemnew/show/P1073 由于任何城市都可以多次经过,所以可以随便走,也就不用太在意有向边和无向边,把无向边当做两条有向边处理: 根 ...

随机推荐

  1. 内部网关协议RIP 路由选择算法(距离向量)

    RIP是一种基于距离向量的路由选择协议 RIP的距离就是指的跳数,没经过一个路由,就是一跳,RIP允许一跳路径最多经过15个路由器,所以16个的话就相当于不可以到达了 RIP协议的特点: 1:仅和相邻 ...

  2. LintCode-50.数组剔除元素后的乘积

    数组剔除元素后的乘积 给定一个整数数组A. 定义B[i] = A[0] * ... * A[i-1] * A[i+1] * ... * A[n-1], 计算B的时候请不要使用除法. 样例 给出A=[1 ...

  3. <Android>日期,时间选择对话框

    a)         调用Activity的onCreateDialog()方法创建对话框 b)        分别在OnDateSetListener的onDateSet()方法和OnTimeSet ...

  4. C# 执行bat文件

    private void RunBat(string batPath) { Process pro = new Process(); FileInfo file = new FileInfo(batP ...

  5. Personal summary 个人总结

    一.请回望开学时的第一次作业,你对于软件工程课程的想象 对比开篇博客你对课程目标和期待,"希望通过实践锻炼,增强计算机专业的能力和就业竞争力",对比目前的所学所练所得,在哪些方面达 ...

  6. 【alpha】Scrum站立会议第3次....10.18

    小组名称:nice! 小组成员:李权 于淼 杨柳 刘芳芳 项目内容:约跑app(约吧--暂定) 1.任务进度 2.燃尽图 功能列表 1.登录注册 2.创建跑步计划 3.筛选跑友 4.加一起跑步的人为好 ...

  7. 如何彻底解决adb 5037端口被占用

    在进行安卓开发的时候是不是经常碰到adb端口被占用的情况? 解决这个问题的方法很简单,只需要设置一个系统环境变量就可以搞定. 设置方法: 增加系统环境变量变量名称:ADNROID_ADB_SERVER ...

  8. (沒有介紹標準算法的)RMQ問題

    感謝杜哥代碼滋磁 //以下是廢話 RMQ (Range Minimum/Maximum Query)问题是指:对于长度为n的数列A,回答若干询问RMQ(A,i,j)(i,j<=n),返回数列A中 ...

  9. [bzoj] 3669 NOI2014 魔法森林 || LCT

    原题 copy一篇题解:原链接 将边按照a排序,然后从小到大枚举,加入图中. 在图中用lct维护一棵两点之间b最大值尽量小的生成树. 当加入一条边(u, v)时: 如果(u, v)不连通,则直接加入这 ...

  10. [usaco] 2008 Dec Largetst Fence 最大的围栏 2 || dp

    原网站大概已经上不了了-- 题目大意: 求出平面上n个点组成的一个包含顶点数最多的凸多边形.n<=250. 考虑我们每次枚举凸包的左下角为谁(参考Graham求凸包时的左下角),然后像Graha ...