HDU 5915 The Fastest Runner Ms. Zhang (CCPC2016 长春 E题,分类讨论 + 求字典序最小的直径 + 数据结构寻找最小值)
题目链接 CCPC2016 Changchun Problem E
题意 给定一个$n$个点$n$条边的无向图,现在从某一点$s$出发,每个点都经过一遍,最后在$t$点停止,经过的边数为$l$
求字典序最小的三元组$(l, s, t)$
设环的长度为$c$,
当$s$和$t$在同一棵子树上的时候,$s$到$t$的路径上的边和环上的边只要走一次,其他边都要走两次,那么答案为$2n - c - len$
$len$为$s$到$t$的路径的长度;
当$s$和$t$不在同一棵子树上的时候,设$s$走到环上的第一个点为$u$,$t$走到环的第一个点为$v$。
那么这个时候考虑每条边走过的次数。
对于不在环上的边,从$s$到$u$,从$v$到$t$的路径只要走一次,其余的边都要走两次。
对于在环上的边,从$u$到$v$的路径只要走一次,其余的部分都要走两次,但是有一条边不用经过(一次都不用走)
那么答案为$2n - 2 - len(s, u) - len(u, v) - len(v, t)$;
对于第一种情况,我们对环上每棵树都求一条字典序最小的直径然后更新答案即可。
对于第二种情况,我们需要最大化$len(s, u) + len(u, v) + len(v, t)$的值。
考虑每个环上的点,显然$s$和$t$的最佳选择都是从这个点往里走可以走到的最深的点。
设环上的点依次排列为:$a_{1}, a_{2}, a_{3}, ......, a_{cnt}$。
设每个环上的点往子树方向走能走到的最深的深度为$c_{1}, c_{2}, c_{3}, ......, c_{cnt}$
倍长a数组和c数组,对于每个位置$i(cnt < i <= 2 * cnt)$,找到一个最佳的点$j(i - cnt + 1 <= j <= i - 1)$。
使得$i - j + c[j] + c[i] = i + c[i] + (c[j] - j)$的值最大。
那么用单调队列维护$c[j] - j$的最大值即可。我为了方便偷懒用了ST表。
- #include <bits/stdc++.h>
- using namespace std;
- #define rep(i, a, b) for (int i(a); i <= (b); ++i)
- #define dec(i, a, b) for (int i(a); i >= (b); --i)
- #define fi first
- #define se second
- #define MP make_pair
- typedef long long LL;
- typedef pair <int, int> PII;
- const int N = 1e5 + 10;
- struct node{
- int x, y;
- } b[N << 1];
- node f[N << 1][20];
- int T;
- int n, cnt;
- int a[N], isroot[N], father[N], vis[N];
- int len, l, r;
- int lg[N << 1];
- int m;
- int ca = 0;
- vector <int> v[N];
- pair <int, PII> ans;
- PII c[N];
- int get_circle(int x){
- vis[x] = 1;
- for (auto u : v[x]){
- if (u == father[x]) continue;
- father[u] = x;
- if (vis[u]){
- cnt = 0;
- int w = x;
- while (w ^ u){
- a[++cnt] = w;
- isroot[w] = cnt;
- w = father[w];
- }
- a[++cnt] = u;
- isroot[u] = cnt;
- return 1;
- }
- if (get_circle(u)) return 1;
- }
- return 0;
- }
- inline node mx(const node &a, const node &b){
- if (a.x == b.x){
- if (a.y < b.y) return a;
- else return b;
- }
- if (a.x > b.x) return a; else return b;
- }
- void dfs(int x, int fa, int dep){
- if ((dep > len) || (dep == len && x < l)){
- len = dep;
- l = x;
- }
- for (auto u : v[x]){
- if (u == fa || isroot[u]) continue;
- dfs(u, x, dep + 1);
- }
- }
- void dfs2(int x, int fa, int dep, int extra){
- if ((dep > len) || (dep == len && x < r)){
- len = dep;
- r = x;
- }
- for (auto u : v[x]) if ((u != fa) && (!isroot[u] || u == extra)){
- dfs2(u, x, dep + 1, extra);
- }
- }
- inline node solve(int l, int r){
- int k = lg[r - l + 1];
- return mx(f[l][k], f[r - (1 << k) + 1][k]);
- }
- int main(){
- lg[1] = 0;
- rep(i, 2, 2e5) lg[i] = lg[i >> 1] + 1;
- scanf("%d", &T);
- while (T--){
- scanf("%d", &n);
- if (n == 2) while (1);
- rep(i, 0, n + 1){
- v[i].clear();
- isroot[i] = 0;
- father[i] = 0;
- vis[i] = 0;
- }
- cnt = 0;
- rep(i, 1, n){
- int x, y;
- scanf("%d%d", &x, &y);
- v[x].push_back(y);
- v[y].push_back(x);
- }
- father[1] = 0;
- get_circle(1);
- memset(c, -1, sizeof c);
- ans = MP(1e9, MP(-1, -1));
- rep(i, 1, cnt){
- len = -1;
- l = -1;
- r = -1;
- dfs(a[i], 0, 0);
- c[i] = MP(len, l);
- len = -1;
- dfs2(l, 0, 0, a[i]);
- if (l > r) swap(l, r);
- ans = min(ans, MP(2 * n - len - cnt, MP(l, r)));
- }
- rep(i, 1, cnt){
- b[i] = {c[i].fi - i + 1, c[i].se};
- b[i + cnt] = {c[i].fi - (i - 1 + cnt), c[i].se};
- }
- m = cnt << 1;
- memset(f, 0, sizeof f);
- rep(i, 1, m) f[i][0] = b[i];
- rep(j, 1, 18){
- rep(i, 1, m){
- if ((i + (1 << j) - 1) <= m) f[i][j] = mx(f[i][j - 1], f[i + (1 << (j - 1))][j - 1]);
- }
- }
- rep(i, cnt + 1, m){
- node now = solve(i - cnt + 1, i - 1);
- int ll = now.y, rr = c[i - cnt].se;
- if (ll > rr) swap(ll, rr);
- int ret = c[i - cnt].fi + i - 1 + (now.x);
- ans = min(ans, MP(2 * n - 2 - ret , MP(ll, rr) ) );
- }
- printf("Case #%d: %d %d %d\n", ++ca, ans.fi, ans.se.fi, ans.se.se);
- }
- return 0;
- }
HDU 5915 The Fastest Runner Ms. Zhang (CCPC2016 长春 E题,分类讨论 + 求字典序最小的直径 + 数据结构寻找最小值)的更多相关文章
- HDU 4423 Simple Function(数学题,2012长春D题)
Simple Function Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)T ...
- hdu 2191 珍惜现在,感恩生活 多重背包入门题
背包九讲下载CSDN 背包九讲内容 多重背包: hdu 2191 珍惜现在,感恩生活 多重背包入门题 使用将多重背包转化为完全背包与01背包求解: 对于w*num>= V这时就是完全背包,完全背 ...
- HDU 2080 夹角有多大II (数学) atan(y/x)分类求角度
夹角有多大II Problem Description 这次xhd面临的问题是这样的:在一个平面内有两个点,求两个点分别和原点的连线的夹角的大小.注:夹角的范围[0,180],两个点不会在圆心出现. ...
- HDU 5203 Rikka with wood sticks 分类讨论
题目链接: hdu:http://acm.hdu.edu.cn/showproblem.php?pid=5203 bc(chinese):http://bestcoder.hdu.edu.cn/con ...
- HDU 1533:Going Home(KM算法求二分图最小权匹配)
http://acm.hdu.edu.cn/showproblem.php?pid=1533 Going Home Problem Description On a grid map there ...
- HDU 6627 equation (分类讨论)
2019 杭电多校 5 1004 题目链接:HDU 6627 比赛链接:2019 Multi-University Training Contest 5 Problem Description You ...
- HDU 6665 Calabash and Landlord (分类讨论)
2019 杭电多校 8 1009 题目链接:HDU 6665 比赛链接:2019 Multi-University Training Contest 8 Problem Description Cal ...
- hdu 1465:不容易系列之一(递推入门题)
不容易系列之一 Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Total Sub ...
- HDU 2504 又见GCD(最大公约数与最小公倍数变形题)
又见GCD Time Limit: 1000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) Total Subm ...
随机推荐
- python2.7写入文件时指定编码为utf-8
python3.0可以这样写 f = open('ufile.log', 'w', 'utf-8') 但在python2.7中open()没有编码参数,如上那样写会报错,可以使用如下模块 impo ...
- appium 多个设备同时执行
测试需要同时在多个android设备上运行,就需要启动多个appium 使用adb命令获取udid,命令:adb get-serialno 使用的是testng测试框架,代码使用java编写 第一台, ...
- == 与 equals 之区别
"=="和equals方法究竟有什么区别? (单独把一个东西说清楚,然后再说清楚另一个,这样,它们的区别自然就出来了,混在一起说,则很难说清楚) ==操作符专门用来比较两个变量的值 ...
- Oz 创建CentOS7镜像
参考链接: https://github.com/clalancette/oz/wiki/Oz-template-description-language https://github.com/cla ...
- 【多线程学习(1)】创建java多线程
1)java多线程的创建方式有三种: 1.继承Thread类 2.实现Runnable接口 3.实现Callable接口 第一种: //继承Thread类 class ExtendsThread ex ...
- C++ 虚继承内存分配
我们知道,虚继承的基类在类的层次结构中只可能出现一个实例.虚基类在类的层次结构中的位置是不能固定的,因为继承了虚基类的类可能会再次被其他类多继承. 比如class A: virtual T{} 这时T ...
- Win10 Ubuntu18.04 编译安装 nignx
nginx 下载页 http://nginx.org/en/download.html wget http://nginx.org/download/nginx-1.14.0.tar.gz //安装依 ...
- 【Python】- 第一行跟第二行的写法
第一行:目的就是指出,你想要你的这个文件中的代码用什么可执行程序去运行它,就这么简单 #!/usr/bin/python:告诉操作系统执行这个脚本的时候,调用/usr/bin下的python解释器: ...
- 【DNS】DNS的几个基本概念
一. 根域 就是所谓的“.”,其实我们的网址www.baidu.com在配置当中应该是www.baidu.com.(最后有一点),一般我们在浏览器里输入时会省略后面的点,而这也已经成为了习惯. 根域服 ...
- macOS Sierra下如何打开任何来源(10.12系统)
转载声明:本站文章无特别说明皆为原创,转载请注明:史蒂芬周的博客, 一定有很多朋友和小子一样,迫不及待的升级到了macOS Sierra,随之而来的是第三方应用都无法打开了,提示无法打开或者扔进废纸篓 ...