【BZOJ4016】[FJOI2014]最短路径树问题

题面

bzoj

洛谷

题解

虽然调了蛮久,但是思路还是蛮简单的2333

把最短路径树构出来,然后点分治就好啦

ps:如果树构萎了,这组数据可以卡掉

代码

#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <algorithm>
#include <queue>
#include <map>
#include <vector>
using namespace std;
inline int gi() {
register int data = 0, w = 1;
register char ch = 0;
while (!isdigit(ch) && ch != '-') ch = getchar();
if (ch == '-') w = -1, ch = getchar();
while (isdigit(ch)) data = 10 * data + ch - '0', ch = getchar();
return w * data;
}
const int MAX_N = 3e4 + 5;
const int MAX_M = 6e4 + 5;
const int INF = 1e9;
int N, M, K;
struct Graph { int to, next, cost; } e[MAX_M << 2];
int fir1[MAX_N], fir2[MAX_N], e_cnt;
void clearGraph() {
memset(fir1, -1, sizeof(fir1));
memset(fir2, -1, sizeof(fir2));
e_cnt = 0;
}
void Add_Edge(int *fir, int u, int v, int w) { e[e_cnt] = (Graph){v, fir[u], w}; fir[u] = e_cnt++; }
vector<int> vec[MAX_N];
bool vis[MAX_N];
bool cmp(int i, int j) { return e[i].to < e[j].to; }
void dijkstra() {
static priority_queue<pair<int, int>, vector<pair<int, int> >, greater<pair<int, int> > > que;
static int dis[MAX_N];
fill(&dis[1], &dis[N + 1], INF);
dis[1] = 0, que.push(make_pair(0, 1));
while (!que.empty()) {
pair<int, int> p = que.top(); que.pop();
int x = p.second;
if (p.first > dis[x]) continue;
for (int i = fir1[x]; ~i; i = e[i].next) {
int v = e[i].to;
if (dis[v] + e[i].cost == dis[x]) vec[v].push_back(i ^ 1);
}
for (int i = fir1[x]; ~i; i = e[i].next) {
int v = e[i].to;
if (dis[x] + e[i].cost < dis[v]) {
dis[v] = dis[x] + e[i].cost;
que.push(make_pair(dis[v], v));
}
}
}
}
void dfs(int x) {
vis[x] = 1;
sort(vec[x].begin(), vec[x].end(), cmp);
for (int i = 0, sz = vec[x].size(); i < sz; i++) {
int j = vec[x][i]; if (vis[e[j].to]) continue;
Add_Edge(fir2, x, e[j].to, e[j].cost), Add_Edge(fir2, e[j].to, x, e[j].cost);
dfs(e[j].to);
}
}
bool used[MAX_N];
int size[MAX_N], dep[MAX_N], dis[MAX_N], centroid, sz, rmx, mx;
int stk[MAX_N], top = 0;
int ans1 = 0, ans2 = 0;
void search_centroid(int x, int fa) {
size[x] = 1;
int mx = 0;
for (int i = fir2[x]; ~i; i = e[i].next) {
int v = e[i].to;
if (v == fa || used[v]) continue;
search_centroid(v, x);
size[x] += size[v];
mx = max(mx, size[v]);
}
mx = max(mx, sz - size[x]);
if (mx < rmx) centroid = x, rmx = mx;
}
namespace cpp1 {
int bln[MAX_N];
void getans(int x, int fa) {
if (dep[x] < K - 1) ans1 = max(ans1, bln[K - 1 - dep[x]] + dis[x]);
for (int i = fir2[x]; ~i; i = e[i].next) {
int v = e[i].to; if (used[v] || v == fa) continue;
dep[v] = dep[x] + 1;
dis[v] = dis[x] + e[i].cost;
getans(v, x);
}
}
void getdis(int x, int fa) {
stk[++top] = x; bln[dep[x]] = max(bln[dep[x]], dis[x]);
for (int i = fir2[x]; ~i; i = e[i].next) {
int v = e[i].to; if (used[v] || v == fa) continue;
getdis(v, x);
}
}
void Div(int x) {
used[x] = 1; top = 0;
for (int i = fir2[x]; ~i; i = e[i].next) {
int v = e[i].to; if (used[v]) continue;
dep[v] = 1, dis[v] = e[i].cost;
getans(v, 0), getdis(v, 0);
}
ans1 = max(ans1, bln[K - 1]);
for (int i = 1; i <= top; i++) bln[dep[stk[i]]] = 0;
for (int i = fir2[x]; ~i; i = e[i].next) {
int v = e[i].to; if (used[v]) continue;
rmx = sz = size[v], centroid = 0;
search_centroid(v, 0);
Div(centroid);
}
}
}
namespace cpp2 {
map<pair<int, int>, int> mp;
void getans(int x, int fa) {
if (dep[x] < K - 1) ans2 += mp[make_pair(ans1 - dis[x], K - 1 - dep[x])];
for (int i = fir2[x]; ~i; i = e[i].next) {
int v = e[i].to; if (used[v] || v == fa) continue;
dep[v] = dep[x] + 1;
dis[v] = dis[x] + e[i].cost;
getans(v, x);
}
}
void getdis(int x, int fa) {
mp[make_pair(dis[x], dep[x])]++;
for (int i = fir2[x]; ~i; i = e[i].next) {
int v = e[i].to; if (used[v] || v == fa) continue;
getdis(v, x);
}
}
void Div(int x) {
used[x] = 1;
for (int i = fir2[x]; ~i; i = e[i].next) {
int v = e[i].to; if (used[v]) continue;
dep[v] = 1, dis[v] = e[i].cost;
getans(v, 0), getdis(v, 0);
}
ans2 += mp[make_pair(ans1, K - 1)];
mp.clear();
for (int i = fir2[x]; ~i; i = e[i].next) {
int v = e[i].to; if (used[v]) continue;
rmx = sz = size[v], centroid = 0;
search_centroid(v, 0);
Div(centroid);
}
}
}
int main () {
N = gi(), M = gi(), K = gi();
clearGraph();
for (int i = 1; i <= M; i++) {
int u = gi(), v = gi(), w = gi();
Add_Edge(fir1, u, v, w), Add_Edge(fir1, v, u, w);
}
dijkstra();
dfs(1);
sz = rmx = N;
search_centroid(1, 0);
cpp1::Div(centroid);
memset(used, 0, sizeof(used));
sz = rmx = N, centroid = 0;
search_centroid(1, 0);
cpp2::Div(centroid);
printf("%d %d\n", ans1, ans2);
return 0;
}

【BZOJ4016】[FJOI2014]最短路径树问题的更多相关文章

  1. [BZOJ4016][FJOI2014]最短路径树问题

    [BZOJ4016][FJOI2014]最短路径树问题 试题描述 给一个包含n个点,m条边的无向连通图.从顶点1出发,往其余所有点分别走一次并返回. 往某一个点走时,选择总长度最短的路径走.若有多条长 ...

  2. [BZOJ4016][FJOI2014]最短路径树问题(dijkstra+点分治)

    4016: [FJOI2014]最短路径树问题 Time Limit: 5 Sec  Memory Limit: 512 MBSubmit: 1796  Solved: 625[Submit][Sta ...

  3. 【BZOJ4016】[FJOI2014]最短路径树问题 最短路径树+点分治

    [BZOJ4016][FJOI2014]最短路径树问题 Description 给一个包含n个点,m条边的无向连通图.从顶点1出发,往其余所有点分别走一次并返回. 往某一个点走时,选择总长度最短的路径 ...

  4. 【BZOJ4016】[FJOI2014]最短路径树问题(点分治,最短路)

    [BZOJ4016][FJOI2014]最短路径树问题(点分治,最短路) 题面 BZOJ 洛谷 题解 首先把最短路径树给构建出来,然后直接点分治就行了. 这个东西似乎也可以长链剖分,然而没有必要. # ...

  5. bzoj 4016 [FJOI2014]最短路径树问题(最短路径树+树分治)

    4016: [FJOI2014]最短路径树问题 Time Limit: 5 Sec  Memory Limit: 512 MBSubmit: 426  Solved: 147[Submit][Stat ...

  6. BZOJ_4016_[FJOI2014]最短路径树问题_最短路+点分治

    BZOJ_4016_[FJOI2014]最短路径树问题_最短路+点分治 Description 给一个包含n个点,m条边的无向连通图.从顶点1出发,往其余所有点分别走一次并返回. 往某一个点走时,选择 ...

  7. 【BZOJ-4016】最短路径树问题 Dijkstra + 点分治

    4016: [FJOI2014]最短路径树问题 Time Limit: 5 Sec  Memory Limit: 512 MBSubmit: 1092  Solved: 383[Submit][Sta ...

  8. [FJOI2014]最短路径树问题 长链剖分

    [FJOI2014]最短路径树问题 LG传送门 B站传送门 长链剖分练手好题. 如果你还不会长链剖分的基本操作,可以看看我的总结. 这题本来出的很没水平,就是dijkstra(反正我是不用SPFA)的 ...

  9. 洛谷 [FJOI2014]最短路径树问题 解题报告

    [FJOI2014]最短路径树问题 题目描述 给一个包含\(n\)个点,\(m\)条边的无向连通图.从顶点\(1\)出发,往其余所有点分别走一次并返回. 往某一个点走时,选择总长度最短的路径走.若有多 ...

随机推荐

  1. 2018-2019-2 《网络对抗技术》Exp0 Kali安装 Week1 20165322

    Exp0 Kali安装 由于VMWare已经装好,镜像也是从上届拷下来的.下载部分就省略.. 新建虚拟机 安装好VM后点击新建虚拟机 选择稍后安装操作系统,系统选择Linux-ubuntu 接下来根据 ...

  2. 随手练——HDU 1251 统计难题

    知识点:前缀树.典型的前缀树模板. 这是用next[26]数组的版本,超内存了.(后来发现,用C++交不会超,G++就会超) #include <iostream> #include &l ...

  3. 连接池中的maxIdle,MaxActive,maxWait等参数详解

    转: 连接池中的maxIdle,MaxActive,maxWait等参数详解 2017年06月03日 15:16:22 阿祥小王子 阅读数:6481   版权声明:本文为博主原创文章,未经博主允许不得 ...

  4. 复习静态页面polo-360

    1.ps快捷键 ctrl+1 恢复到100% ctrl+0 适应屏幕大小 ctrl+r 显示标尺 辅助线的利用 矩形框--图像--裁剪:文件存储为web所用格式,注意选格式. 1个像素的平铺 雪碧图的 ...

  5. StackExchange.Redis学习笔记(一) Redis的使用初探

    Redis Redis将其数据库完全保存在内存中,仅使用磁盘进行持久化. 与其它键值数据存储相比,Redis有一组相对丰富的数据类型. Redis可以将数据复制到任意数量的从机中 Redis的安装 官 ...

  6. 从零搭建vue

    第一步: 安装node.js,一般安装 长期维护版  相对比较稳定 点击下载,下载好了之后双击运行,可选择安装路径,然后一路下一步即可. 安装完成后,在cmd输入 node -v 如果出现版本号,则安 ...

  7. [Python]json对象转换出错expected string or buffer python

    [问题] 今天在使用python中的json转换碰到一个问题: 错误显示: expected string or buffer json内容如下: {u'err_no': 0, u'corpus_no ...

  8. POJ 3762 The Bonus Salary!(最小K覆盖)

    POJ 3762 The Bonus Salary! 题目链接 题意:给定一些任务.每一个任务有一个时间,有k天.一个时间仅仅能运行一个任务,每一个任务有一个价值.问怎么安排能得到最多价值 思路:典型 ...

  9. ubuntu 如何进行文件、夹删除等操作

    rm [选项] 文件-f, --force 强力删除,不要求确认-i 每删除一个文件或进入一个子目录都要求确认-I 在删除超过三个文件或者递归删除前要求确认-r, -R 递归删除子目录-d, --di ...

  10. 『ACM C++』 PTA 天梯赛练习集L1 | 027-028

    死亡周二,今天去看惊奇队长了!!!真的很佩服国外的后期特效大片技术,要是我们国内也能实现这样的技术能力就好了~ 羡慕max -------------------------------------- ...