luogu2149 [SDOI2009] Dlaxia的路线
题目大意
在一个无向图中,定义两个点s,t的最短路径子图为一个极大边集,对于该边集内的所有有向边e,总存在一条起点为s,终点为t且经过边e的路径,使得该路径长度为s到t的最短路径长度。现给出一个无向图,s1, s2, t1, t2四个节点,求一条最长的路径,使得它满足下列条件之一:1. 该路径上的所有边都属于s1到t1的最短路径子图且属于s2到t2的最短路径子图;2. 该路径上的所有边都属于s1到t1的最短路径子图且属于t2到s2的最短路径子图。输出该路径的长度。
题解
我们可以用4次Dijkstra得到s1,t1和s2,t2的最短路径子图$G_1,G_2$,另外由$G_2$可以得到其反向图$G'_2$。然后分别在子图$G_1\cap G_2$和$G_1\cap G'_2$上进行拓扑排序求最长路径即可。本题最重要的就是从题面到数学语言的翻译过程了,如果这一点不明确,我们可能就会建立一个子图$G_1\cap(G_2\cup G'_2)$,这样就乱了。
#include <cstdio>
#include <cstring>
#include <algorithm>
#include <queue>
#include <stack>
#include <cassert>
using namespace std; const int MAX_NODE = 1510, MAX_EDGE = 200005, INF = 0x3f3f3f3f; struct Node;
struct Edge; struct Node
{
int DistS, DistT;
bool Done;
int Dist;//Dijkstra时,它是临时的东东;TopSort时,它是真正的最长路径长度
int DfsN;
Edge *Head;
}_nodes[MAX_NODE];
int TotNode; struct HeapNode
{
Node *cur;
int Dist; HeapNode(Node *x):cur(x),Dist(x->Dist){} bool operator < (const HeapNode& a) const
{
return Dist > a.Dist;
}
}; struct Edge
{
int Weight;
bool InSPG;//在第一个人的最短路径子图中
bool InTopG;//在Top子图中
Node *To, *From;
Edge *Next, *Rev; Edge() :InTopG(false), InSPG(false){}
}_edges[MAX_EDGE];
vector<Edge*> Temp;
int _eCount; Edge *NewEdge()
{
if (_eCount < MAX_EDGE - 1)
return _edges + ++_eCount;
else
{
Temp.push_back(new Edge());
return Temp.back();
}
} Edge *AddEdge(Node *from, Node *to, int w)
{
Edge *e = NewEdge();
e->From = from;
e->To = to;
e->Weight = w;
e->Next = from->Head;
from->Head = e;
return e;
} void Build(int uId, int vId, int w)
{
Node *u = _nodes + uId, *v = _nodes + vId;
Edge *e1 = AddEdge(u, v, w), *e2 = AddEdge(v, u, w);
e1->Rev = e2;
e2->Rev = e1;
} void Dijkstra(Node *start)
{
static priority_queue<HeapNode> q;
for (int i = 1; i <= TotNode; i++)
{
_nodes[i].Dist = INF;
_nodes[i].Done = false;
}
start->Dist = 0;
q.push(start);
while (!q.empty())
{
HeapNode temp = q.top();
q.pop();
Node *cur = temp.cur;
if (cur->Done)
continue;
cur->Done = true;
for (Edge *e = cur->Head; e; e = e->Next)
{
if (cur->Dist + e->Weight < e->To->Dist)
{
e->To->Dist = cur->Dist + e->Weight;
q.push(e->To);
}
}
}
} void GetInGraph(void(*DoInGraph)(Edge*), int spLen)
{
for (int i = 1; i <= _eCount; i++)
if (_edges[i].From->DistS + _edges[i].Weight + _edges[i].To->DistT == spLen)
DoInGraph(_edges + i);
for (int i = 0; i < Temp.size(); i++)
if (Temp[i]->From->DistS + Temp[i]->Weight + Temp[i]->To->DistT == spLen)
DoInGraph(Temp[i]);
} void SetOrgGraph(Edge *e)
{
e->InSPG = true;
} void MakeTopGraph1(Edge *e)
{
if (e->InSPG)
e->InTopG = true;
} void MakeTopGraph2(Edge *e)
{
if (e->Rev->InSPG)
e->InTopG = true;
} int ShortestPath(int s, int t)
{
Node *start = _nodes + s, *target = _nodes + t;
Dijkstra(target);
for (int i = 1; i <= TotNode; i++)
_nodes[i].DistT = _nodes[i].Dist;
Dijkstra(start);
for (int i = 1; i <= TotNode; i++)
_nodes[i].DistS = _nodes[i].Dist;
return target->DistS;
} void ClearTopGraph()
{
for (int i = 1; i <= _eCount; i++)
_edges[i].InTopG = false;
for (int i = 0; i < Temp.size(); i++)
Temp[i]->InTopG = false;
} stack<Node*> St;
void Dfs(Node *cur)
{
assert(cur->DfsN != 1);
if (cur->DfsN == 2)
return;
cur->DfsN = 1;
for (Edge *e = cur->Head; e; e = e->Next)
{
if (!e->InTopG)
continue;
Dfs(e->To);
}
cur->DfsN = 2;
St.push(cur);
} int LongestPath()
{
for (int i = 1; i <= TotNode; i++)
{
_nodes[i].Dist = 0;
_nodes[i].DfsN = 0;
}
for (int i = 1; i <= TotNode; i++)
Dfs(_nodes + i);
int ans = 0;
while (!St.empty())
{
Node *cur = St.top();
St.pop();
ans = max(ans, cur->Dist);
for (Edge *e = cur->Head; e; e = e->Next)
if (e->InTopG)
e->To->Dist = max(e->To->Dist, cur->Dist + e->Weight);
}
return ans;
} int main()
{
int totEdge, s1, s2, t1, t2;
scanf("%d%d%d%d%d%d", &TotNode, &totEdge, &s1, &t1, &s2, &t2);
for (int i = 1; i <= totEdge; i++)
{
int u, v, w;
scanf("%d%d%d", &u, &v, &w);
Build(u, v, w);
}
int len1 = ShortestPath(s1, t1);
GetInGraph(SetOrgGraph, len1);
int len2 = ShortestPath(s2, t2);
GetInGraph(MakeTopGraph1, len2);
int ans1 = LongestPath();
ClearTopGraph();
GetInGraph(MakeTopGraph2, len2);
int ans2 = LongestPath();
printf("%d\n", max(ans1, ans2));
return 0;
}
luogu2149 [SDOI2009] Dlaxia的路线的更多相关文章
- Luogu2149 [SDOI2009]Elaxia的路线-最短路+拓扑排序
Solution 另外$ m <=5e5$. 两条最短路的 最长公共路径 一定是若干条连续的边, 并且满足拓扑序. 于是我们分别 正向 和反向走第二条路径,若该条边同时是两条最短路径上的边, 则 ...
- BZOJ 1880: [Sdoi2009]Elaxia的路线( 最短路 + dp )
找出同时在他们最短路上的边(dijkstra + dfs), 组成新图, 新图DAG的最长路就是答案...因为两人走同一条路但是不同方向也可以, 所以要把一种一个的s,t换一下再更新一次答案 ---- ...
- 【BZOJ1880】[Sdoi2009]Elaxia的路线(最短路)
[BZOJ1880][Sdoi2009]Elaxia的路线(最短路) 题面 BZOJ 洛谷 题解 假装我们知道了任意两点间的最短路,那么我们怎么求解答案呢? 不难发现公共路径一定是一段连续的路径(如果 ...
- 洛谷 P2149 [SDOI2009]Elaxia的路线 解题报告
P2149 [SDOI2009]Elaxia的路线 题目描述 最近,Elaxia和w**的关系特别好,他们很想整天在一起,但是大学的学习太紧张了,他们 必须合理地安排两个人在一起的时间. Elaxia ...
- 【BZOJ 1880】 [Sdoi2009]Elaxia的路线 (最短路树)
1880: [Sdoi2009]Elaxia的路线 Description 最近,Elaxia和w**的关系特别好,他们很想整天在一起,但是大学的学习太紧张了,他们 必须合理地安排两个人在一起的时间. ...
- BZOJ1880: [Sdoi2009]Elaxia的路线(最短路)
1880: [Sdoi2009]Elaxia的路线 Time Limit: 4 Sec Memory Limit: 64 MBSubmit: 2049 Solved: 805 题目链接:https ...
- 【BZOJ1880】[SDOI2009]Elaxia的路线 (最短路+拓扑排序)
[SDOI2009]Elaxia的路线 题目描述 最近,\(Elaxia\)和\(w**\)的关系特别好,他们很想整天在一起,但是大学的学习太紧张了,他们 必须合理地安排两个人在一起的时间. \(El ...
- 【BZOJ1880】[Sdoi2009]Elaxia的路线 最短路+DP
[BZOJ1880][Sdoi2009]Elaxia的路线 Description 最近,Elaxia和w**的关系特别好,他们很想整天在一起,但是大学的学习太紧张了,他们 必须合理地安排两个人在一起 ...
- 洛谷——P2149 [SDOI2009]Elaxia的路线
P2149 [SDOI2009]Elaxia的路线 题目描述 最近,Elaxia和w的关系特别好,他们很想整天在一起,但是大学的学习太紧张了,他们 必须合理地安排两个人在一起的时间.Elaxia和w每 ...
随机推荐
- [ HAOI 2010 ] 最长公共子序列
\(\\\) \(Description\) 求两个长度\(\le5000\)的大写字母串的\(LCS\)长度及个数,定义两\(LCS\)中某一字符在两序列出现位置有一处不同就视为不同. \(\\\) ...
- TensorFlow-Gpu环境搭建——Win10+ Python+Anaconda+cuda
参考:http://blog.csdn.net/sb19931201/article/details/53648615 https://segmentfault.com/a/1190000009803 ...
- JS——属性绑定
1.普通形式 <script> var stu = new Object(); stu.name = "ww"; console.log(stu);//{name: & ...
- Python 之mysql类封装
import pymysql class MysqlHelper(object): conn = None def __init__(self, host, username, password, d ...
- Vue中this.$router.push参数获取(通过路由传参)【路由跳转的方法】
传递参数的方法: 1.Params 由于动态路由也是传递params的,所以在 this.$router.push() 方法中 path不能和params一起使用,否则params将无效.需要用nam ...
- ImageMagick的下载和配置
2. 新建一个VC++的工程,项目->属性 VC++目录中 包含目录中加入4项,ImageMagick安装路径下的include文件夹,和include里边的三个文件夹. 库目录中加入1项,Im ...
- Ansible实现zabbix服务器agent端批量部署
项目需求:由于搭建zabbix,需要每台服务器都需要安装监控端(agent)正常的的操作是一台一台去安装,这样确实有点浪费时间,这里为大家准备了一款开源 的自动化运维工具Ansible,相信大家也很熟 ...
- [forward]警惕UNIX下的LD_PRELOAD环境变量
From: https://blog.csdn.net/haoel/article/details/1602108 警惕UNIX下的LD_PRELOAD环境变量 前言 也许这个话题并不新鲜,因为LD_ ...
- Python学习【第4篇】:Python之文件操作
文件操作 读取一行 f=open("D:\\1.txt",'rb') print f.readline() f.close() 将文件内容保存在一个list with open(& ...
- Python那点事
目录 Python那点事 一. Python介绍 二. 目前python的主要领域 三.Python的运用 四. Python的发展史 Python的种类 Python那点事 相信很多小伙伴会跳过此环 ...