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的更多相关文章

  1. hdu-6166(最短路+二进制分组)

    题意:给你n个点m条边的有向图,然后再给你k个不同的点,问你这k个点的最小距离: 解题思路:这道题最需要注意的就是k个点一定是不同的,那么有一个结论就是任意两个不同的数字中,在他们的二进制地表示中,一 ...

  2. hdu6166 Senior Pan

    Senior Pan Time Limit: 12000/6000 MS (Java/Others)    Memory Limit: 131072/131072 K (Java/Others)Tot ...

  3. 【最短路】【dijkstra】【二进制拆分】hdu6166 Senior Pan

    题意:给你一张带权有向图,问你某个点集中,两两结点之间的最短路的最小值是多少. 其实就是dijkstra,只不过往堆里塞边的时候,要注意塞进去它是从集合中的哪个起始点过来的,然后在更新某个点的答案的时 ...

  4. 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 ...

随机推荐

  1. FFT多项式乘法模板

    有时间来补算法原理orz #include <iostream> #include <cstdio> #include <cmath> #include <c ...

  2. [洛谷P2604][ZJOI2010]网络扩容

    题目大意:给定一张有向图,每条边都有一个容量C和一个扩容费用W.这里扩容费用是指将容量扩大1所需的费用. 求: 1.在不扩容的情况下,1到N的最大流: 2.将1到N的最大流增加K所需的最小费用. 题解 ...

  3. [Leetcode] Recover binary search tree 恢复二叉搜索树

    Two elements of a binary search tree (BST) are swapped by mistake. Recover the tree without changing ...

  4. [Leetcode] Remove duplicates from sorted array 从已排序的数组中删除重复元素

    Given a sorted array, remove the duplicates in place such that each element appear only once and ret ...

  5. wait for it

  6. BZOJ_day4&&DSFZ_day1

    昨天坐火车才水了三道题... 25题 100810221041105110591087108811791191119212571303143218541876195119682140224224382 ...

  7. [hdu 3949]线性基+高斯消元

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3949 一开始给做出来的线性基wa了很久,最后加了一步高斯消元就过了. 之所以可以这样做,证明如下. 首 ...

  8. 湖南大学第十四届ACM程序设计新生杯 Dandan's lunch

    Dandan's lunch Description: As everyone knows, there are now n people participating in the competiti ...

  9. codeforces 1060 D

    https://codeforces.com/contest/1060/problem/D 题意:你可以用1个及以上的圆桌,给n个人排座位,每个人左边需要有Li个空凳子,右边需要有Ri个空凳子,问你最 ...

  10. BZOJ1798 维护序列seq

    1798: [Ahoi2009]Seq 维护序列seq Time Limit: 30 Sec  Memory Limit: 64 MBSubmit: 8058  Solved: 2964[Submit ...