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每 ...
随机推荐
- php入门学习笔记
学习笔记[6.5-6.13] 1.常用命令 打开数据库格式: mysql -h主机地址 -u用户名 -p 重启nginx:sudo /etc/init.d/nginx restart或者service ...
- https Full Handshake
1)加密套件交互: 2)密码交换: 3)身份认证: Full Handshake Initially, client and server "agree upon" null en ...
- C# 获得Properties下的定义的资源
var str1 = Properties.Resources.ResourceManager.GetObject("String1", null); string url = S ...
- JDK升级
保存jboss运行时环境的配置 删除jboss下面的缓存文件 删除deployments里面的war包 重新build项目
- Java面向对象多态
Java面向对象多态 7.2对象 7.2.1创建对象 对象是类的实例 类的名称 对象名称 = new 类的名称(); 7.3继承 继承就是子类继承父类的特征和行为,使得子类对象(实例)具有父类的实例域 ...
- 关于javascript原型链的记录
构造函数拥有名为prototype属性,每个对象都拥有__proto__属性,而且每个对象的__proto__属性指向自身构造函数prototype. **当调用某种方法或属性时,首先会在自身调用或查 ...
- Python MongoDB 教程
基于菜鸟教程实际操作后总结而来 Python MongoDB MongoDB 是目前最流行的 NoSQL 数据库之一,使用的数据类型 BSON(类似 JSON). MongoDB 数据库安装与介绍可以 ...
- Linux系统学习之 一:新手必须掌握的Linux命令1
2018-10-03 16:04:12 一.常用系统工作命令 1.wget 命令 作用:用于在终端中下载网络文件. 格式:wget [参数] 下载地址 参数及作用: -b : 后台下载模式 -d:显示 ...
- 用fallocate进行"文件预留"或"文件打洞"【转】
转自uestc-leon的博客 内容作了一些修改,查看原文请访问uestc-leon 1. 什么是空洞文件? "在UNIX文件操作中,文件位移量可以大于文件的当前长度,在这种情况下,对该文件 ...
- springcloud(四):Eureka客户端公共组件打包方式
, 一.前言 各位大佬应该知道,在大型项目中都需要有数据传输层,一般项目都采用的是MVC结构,如果有10个表,则会创建10个实体类,在各个层之间应该使用实体类传递数据: 在微服架构中,也许 ...