P5304 [GXOI/GZOI2019]旅行者
这里是官方题解
一个图 \(n\) 点 \(m\) 条边,里面有 \(k\) 个特殊点,问这 \(k\) 个点之间两两最短路的最小值是多少?
\(n \leq 10^5, m \leq 5 * 10 ^5\)
假设我们把特殊点分成 \(A,B\) 两个集合,新建 \(s\) 连 \(A\) 集合的所有点,边权 \(0\) ,新建 \(t\) 连接 \(B\) 集合里的所有点,边权 \(0\) ,那么 \(s\) 到 \(t\) 的最短路就是 \(A,B\) 集合点之间的最短路的最小值。
那么对于 \(k\) 个特殊点,我们枚举二进制里的第 \(i\) 位,把二进制第 \(i\) 位是 \(0\) 的点放在 \(A\) , \(1\) 的点放在 \(B\) ,用以上方法跑一个最短路。
然后跑 \(log\ n\) 次最短路之后,所有最短路的最小值就是最终答案。
原理是,假设 \(k\) 个特殊点里最近的是 \(x\) 和 \(y\) ,那么 \(x\) 和 \(y\) 一定有一个二进制位不一样,那么他们肯定在那次分组的时候被放进了不同的集合,从而肯定被算进了最后的答案之中最短路。
#include <bits/stdc++.h>
const int MAXN = 100010, MAXM = 700010;
struct Edge {
int y, z;
Edge *next;
}*a[MAXN], POOL[MAXM], *ptr, *back[MAXN];
void AddEdge(int x, int y, int z) {
Edge *tmp = ptr++;
tmp->y = y;
tmp->z = z;
tmp->next = a[x];
a[x] = tmp;
}
int n, nodes[MAXN], k, s, t;
long long dis[MAXN];
long long dijkstra() {
memset(dis, 0x3f, sizeof(dis));
dis[s] = 0;
std::priority_queue<std::pair<long long, int> > Q;
Q.push(std::make_pair(0, s));
for (int i = 1; i < n + 2; i++) {
while (!Q.empty() && dis[Q.top().second] != -Q.top().first) Q.pop();
if (Q.empty()) break;
int now = Q.top().second;
Q.pop();
for (Edge *p = a[now]; p; p = p->next)
if (dis[p->y] > dis[now] + p->z)
Q.push(std::make_pair(-(dis[p->y] = dis[now] + p->z), p->y));
}
return dis[t];
}
int main(int argc, char* argv[]) {
int T;
scanf("%d", &T);
while (T--) {
ptr = POOL;
memset(a, 0, sizeof a);
int m;
scanf("%d%d%d", &n, &m, &k);
while (m--) {
int x, y, z;
scanf("%d%d%d", &x, &y, &z);
AddEdge(x, y, z);
}
for (int i = 1; i <= k; i++) scanf("%d", nodes + i);
long long Ans = ~0ull>>1;
s = n + 1, t = n + 2;
for (int i = 0; (1 << i) <= k; i++) {
Edge *backup = ptr;
memcpy(back, a, (sizeof a[0]) * (n + 3));
for (int j = 1; j <= k; j++) if (j & (1 << i)) {
AddEdge(s, nodes[j], 0);
} else {
AddEdge(nodes[j], t, 0);
}
Ans = std::min(Ans, dijkstra());
ptr = backup;
memcpy(a, back, (sizeof a[0]) * (n + 3));
for (int j = 1; j <= k; j++) if (j & (1 << i)) {
AddEdge(nodes[j], t, 0);
} else {
AddEdge(s, nodes[j], 0);
}
Ans = std::min(Ans, dijkstra());
ptr = backup;
memcpy(a, back, (sizeof a[0]) * (n + 3));
}
printf("%lld\n", Ans);
}
return 0;
}
P5304 [GXOI/GZOI2019]旅行者的更多相关文章
- 洛谷 P5304 [GXOI/GZOI2019]旅行者(最短路)
洛谷:传送门 bzoj:传送门 参考资料: [1]:https://xht37.blog.luogu.org/p5304-gxoigzoi2019-lv-xing-zhe [2]:http://www ...
- luogu P5304 [GXOI/GZOI2019]旅行者
传送门 所以这个\(5s\)是SMG 暴力是枚举每一个点跑最短路,然后有一个很拿衣服幼稚的想法,就是把所有给出的关键点当出发点,都丢到队列里,求最短路的时候如果当前点\(x\)某个相邻的点\(y\)是 ...
- P5304 [GXOI/GZOI2019]旅行者(最短路/乱搞)
luogu bzoj Orz自己想出神仙正解的sxy 描述略 直接把所有起点推进去跑dijkstra... 并且染色,就是记录到这个点的最短路是由哪个起点引导出来的 然后再把所有边反指跑一次... 之 ...
- 【题解】Luogu P5304 [GXOI/GZOI2019]旅行者
原题传送门 题意:给你k个点,让你求两两最短路之间的最小值 我们考虑二进制拆分,使得每两个点都有机会分在不同的组\((A:0,B:1)\)中,从源点\(S\)向\(A/B\)中的点连边权为0的边,从\ ...
- [洛谷P5304][GXOI/GZOI2019]旅行者
题目大意: 有一张 \(n(n\leqslant10^5)\) 个点 \(m(m\leqslant5\times10^5)\) 条边的有向有正权图,有$k(2\leqslant k\leqslant ...
- [LOJ3087][GXOI/GZOI2019]旅行者——堆优化dijkstra
题目链接: [GXOI/GZOI2019]旅行者 我们考虑每条边的贡献,对每个点求出能到达它的最近的感兴趣的城市(设为$f[i]$,最短距离设为$a[i]$)和它能到达的离它最近的感兴趣的城市(设为$ ...
- 【BZOJ5506】[GXOI/GZOI2019]旅行者(最短路)
[BZOJ5506][GXOI/GZOI2019]旅行者(最短路) 题面 BZOJ 洛谷 题解 正着做一遍\(dij\)求出最短路径以及从谁转移过来的,反过来做一遍,如果两个点不由同一个点转移过来就更 ...
- 洛谷 P 5 3 0 4 [GXOI/GZOI2019]旅行者
题目描述 J 国有 n 座城市,这些城市之间通过 m 条单向道路相连,已知每条道路的长度. 一次,居住在 J 国的 Rainbow 邀请 Vani 来作客.不过,作为一名资深的旅行者,Vani 只对 ...
- BZOJ5506 GXOI/GZOI2019旅行者(最短路)
本以为是个二进制分组傻逼题https://www.cnblogs.com/Gloid/p/9545753.html,实际上有神仙的一个log做法https://www.cnblogs.com/asul ...
随机推荐
- JS 设计模式八 -- 发布订阅者模式
概念 发布---订阅模式又叫观察者模式,它定义了对象间的一种一对多(一个发布,多个观察)的关系,让多个观察者对象同时监听某一个主题对象,当一个对象发生改变时,所有依赖于它的对象都将得到通知. 优点 1 ...
- php密码对称encrypt加密
/** * 对用户的密码进行加密 * @param $password * @param $encrypt //传入加密串,在修改密码时做认证 * @return array/password */ ...
- pstree:command not found
centos7默认并没有安装pstree,所以会有pstree:command not found yum -y install psmisc
- macbook 入门
前面的话 第一次使用 Mac 之前,需要改变一些原有思维,不应该使用 Windows 的思维习惯去使用 Mac,Mac 会节省系统维护.清理杀毒.升级驱动等操作的时间,让我们可以专注做真正重要的事情, ...
- Hadoop系列(二):Hadoop单节点部署
环境:CentOS 7 JDK: 1.7.0_80 hadoop:2.8.5 hadoop(192.168.56.101) 配置基础环境 1. 测试环境可以直接关闭selinux和防火墙 2. 主机添 ...
- Hadoop系列(一):Hadoop集群搭建
环境:CentOS 7 JDK: 1.7.0_80 hadoop:2.8.5 两台机器:master(192.168.56.101) slave(192.168.56.102) 配置基础环境 1. ...
- fast ai-lesson 1 报错解决方法(正则表达式提取文件名)
在运行fast ai lesson 1的代码的时候,运行到的时候报错了 data = ImageDataBunch.from_name_re(path_img, fnames, pat, ds_tfm ...
- vue stylus 格式化问题
IDE是vscode 安装了.vetur插件 由于stylus可以仅用缩进不用写大括号之类的,所以十分方便, 但有个问题,按alt shift F 格式化时,vetur这个插件会默认添加上正常css的 ...
- BZOJ2287 消失之物
这题貌似是个权限题qwq,我是用离线题库+本地数据包测的 题目大意: 给你\(n\)个体积分别为\(w[i]\)的物品和容积\(m\),问你将每一件物品分别去掉之后,拼出\(1\)~\(m\)中每一个 ...
- 安装nova后解决登录没账号问题
找到laravel安装目录执行php artisan nova:user