hdu6166
hdu6166
题意
给出一个有向图,选择 \(k\) 个点,问这 \(k\) 个点任意两点距离的最小值。
分析
按结点编号的二进制位,每次可以把所有点分到两个集合,那么求两个集合的点间的最短路即可( \(0\)作为源点,\(n+1\)作为汇点)。
正确性的保证:编号的唯一性。任意两点一定存在某一次不在同一集合,二进制位一定有某一位不同。
code
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
typedef pair<int, ll> P;
const int MAXN = 1e5 + 10;
const ll INF = 1e18 + 9;
int n, m;
struct node {
int to, cost;
node() {}
node(int to, int cost) : to(to), cost(cost) {}
};
vector<node> G[MAXN];
ll d[MAXN];
ll dijkstra() {
fill(d, d + MAXN, INF);
priority_queue<P, vector<P>, greater<P> > q;
d[0] = 0;
q.push(P(0, 0));
while(!q.empty()) {
P u = q.top(); q.pop();
if(d[u.second] < u.first) continue;
for(int i = 0; i < G[u.second].size(); i++) {
node nd = G[u.second][i];
if(d[nd.to] > d[u.second] + nd.cost) {
d[nd.to] = d[u.second] + nd.cost;
q.push(P(d[nd.to], nd.to));
}
}
}
return d[n + 1];
}
int se[MAXN];
int main() {
int T, kase = 1;
scanf("%d", &T);
while(T--) {
for(int i = 0; i <= n; i++) G[i].clear();
scanf("%d%d", &n, &m);
for(int i = 0; i < m; i++) {
int u, v, c;
scanf("%d%d%d", &u, &v, &c);
G[u].push_back(node(v, c));
}
int k;
scanf("%d", &k);
for(int i = 0; i < k; i++) {
scanf("%d", &se[i]);
}
ll ans = INF;
for(int i = 0; i < 18; i++) {
G[0].clear();
for(int j = 0; j < k; j++) {
if((se[j] >> i) & 1) {
G[0].push_back(node(se[j], 0));
} else {
G[se[j]].push_back(node(n + 1, 0));
}
}
ans = min(ans, dijkstra());
for(int j = 0; j < k; j++) {
if(!((se[j] >> i) & 1)) {
G[se[j]].pop_back();
}
}
G[0].clear();
for(int j = 0; j < k; j++) {
if((se[j] >> i) & 1) {
G[se[j]].push_back(node(n + 1, 0));
} else {
G[0].push_back(node(se[j], 0));
}
}
ans = min(ans, dijkstra());
for(int j = 0; j < k; j++) {
if((se[j] >> i) & 1) {
G[se[j]].pop_back();
}
}
}
printf("Case #%d: %lld\n", kase++, ans);
}
return 0;
}
hdu6166的更多相关文章
- hdu-6166(最短路+二进制分组)
题意:给你n个点m条边的有向图,然后再给你k个不同的点,问你这k个点的最小距离: 解题思路:这道题最需要注意的就是k个点一定是不同的,那么有一个结论就是任意两个不同的数字中,在他们的二进制地表示中,一 ...
- hdu6166 Senior Pan
Senior Pan Time Limit: 12000/6000 MS (Java/Others) Memory Limit: 131072/131072 K (Java/Others)Tot ...
- 【最短路】【dijkstra】【二进制拆分】hdu6166 Senior Pan
题意:给你一张带权有向图,问你某个点集中,两两结点之间的最短路的最小值是多少. 其实就是dijkstra,只不过往堆里塞边的时候,要注意塞进去它是从集合中的哪个起始点过来的,然后在更新某个点的答案的时 ...
- Codeforces 938.D Buy a Ticket
D. Buy a Ticket time limit per test 2 seconds memory limit per test 256 megabytes input standard inp ...
随机推荐
- FFT多项式乘法模板
有时间来补算法原理orz #include <iostream> #include <cstdio> #include <cmath> #include <c ...
- [洛谷P2604][ZJOI2010]网络扩容
题目大意:给定一张有向图,每条边都有一个容量C和一个扩容费用W.这里扩容费用是指将容量扩大1所需的费用. 求: 1.在不扩容的情况下,1到N的最大流: 2.将1到N的最大流增加K所需的最小费用. 题解 ...
- [Leetcode] Recover binary search tree 恢复二叉搜索树
Two elements of a binary search tree (BST) are swapped by mistake. Recover the tree without changing ...
- [Leetcode] Remove duplicates from sorted array 从已排序的数组中删除重复元素
Given a sorted array, remove the duplicates in place such that each element appear only once and ret ...
- wait for it
- BZOJ_day4&&DSFZ_day1
昨天坐火车才水了三道题... 25题 100810221041105110591087108811791191119212571303143218541876195119682140224224382 ...
- [hdu 3949]线性基+高斯消元
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3949 一开始给做出来的线性基wa了很久,最后加了一步高斯消元就过了. 之所以可以这样做,证明如下. 首 ...
- 湖南大学第十四届ACM程序设计新生杯 Dandan's lunch
Dandan's lunch Description: As everyone knows, there are now n people participating in the competiti ...
- codeforces 1060 D
https://codeforces.com/contest/1060/problem/D 题意:你可以用1个及以上的圆桌,给n个人排座位,每个人左边需要有Li个空凳子,右边需要有Ri个空凳子,问你最 ...
- BZOJ1798 维护序列seq
1798: [Ahoi2009]Seq 维护序列seq Time Limit: 30 Sec Memory Limit: 64 MBSubmit: 8058 Solved: 2964[Submit ...