思路

可以考虑另一种想法:因为我们发现,答案是肯定不会大于在原来的树上的最短路径的。所以原来的最短路是(有可能的)最大值!

我们把树变成这样,提取出1~n的路径,方便观看撕烤:

(它有个我起的名字,叫灯笼树态,处理树上路径与其余部分之间的关系时画图用,并非真实的数据结构)

我们称被提取出来的树上路径(本题为1~n)为灯笼线,路径上每一个节点及其下悬挂的子树称为灯笼子树,每一个灯笼子树的根节点叫悬挂点

显然,悬挂点一定在灯笼线上。任意一个点的悬挂点就是它所在的灯笼子树的悬挂点。一个点如果在灯笼线上,那么它的悬挂点是自己。

那么什么时候原来的最短路不是答案呢?

那么就是对于任何的u和v,将它们连接起来后,新的最短路都比原来的最短路小。

换句话说:设灯笼线上的两个点x和y分别为准备新加边的u,v的悬挂点。那么只需比较(x到y的路径长度)和(ux,vy的长度和再加上每次询问时的那个值)即可。

(重头戏来了!)

如果上文的u和v再同一颗灯笼子树内呢?

(*)于是,你发现一个结论:如果一颗灯笼子树除悬挂点外有超过两个以上的节点,那么连接这颗灯笼子树内的两个没有直接边相连的点,无论询问多少,Ans会取到原图的最短路,也就是理论最大值。

所以,我们仅需考虑每颗颗灯笼子树除悬挂点外仅有1的点的原树即可。

由于要比较(x到y的路径长度)和(ux,vy的长度和再加上每次询问时的那个值),所以我们可以算出最大的(x到y的路径长度)-(ux,vy的长度和),每次询问将其加上询问值并与0比较即可。

如何算出(x到y的路径长度)-(ux,vy的长度和)?将这个式子拆开分别维护,一遍dfs,用heap维护一下。建议读者自行撕烤这部分内容。

Code:

#include <cstdio>
#include <queue>
using namespace std;
const int S=600006;
struct info
{
int x;
long long s;
inline bool operator<(const info &a) const
{
return s<a.s;
}
}st[S];
priority_queue<info> pq;
int n,m,h[S],nx[S],v[S],w[S],eg=1,s[S],fa[S],top,lc[S],rc[S];
const long long infl=(1ll<<60)-1;
long long dep[S],res=-infl;
bool flag,mk[S];
void read(int &s)
{
s=0;char c=getchar();
while (c<'0' || c>'9') c=getchar();
while (c>='0' && c<='9') s=(s<<1)+(s<<3)+(c^48),c=getchar();
}
inline void egadd(int uu,int vv,int ww)
{
nx[++eg]=h[uu];h[uu]=eg;
v[eg]=vv;w[eg]=ww;
}
void dfs_1(int x)
{
s[x]=1;
for (int i=h[x];i;i=nx[i])
if (v[i]!=fa[x])
{
fa[v[i]]=x;
dep[v[i]]=dep[x]+w[i];
dfs_1(v[i]);
s[x]+=s[v[i]];
}
}
bool check()
{
int o=n;
if (s[o]>2) return false;
while (o!=1)
{
if (s[fa[o]]-s[o]>2) return false;
mk[o]=true;
o=fa[o];
}
mk[1]=true;
return true;
}
inline long long ma(long long a,long long b){return a>b?a:b;}
inline long long mi(long long a,long long b){return a<b?a:b;}
void dfs_3(int x)//这里是将原树转为二叉树,因为灯笼子树大小不超过两个点的树就是一颗二叉树
{
for (int i=h[x];i;i=nx[i])
if (v[i]!=fa[x])
{
dfs_3(v[i]);
if (mk[v[i]])
{
lc[x]=v[i];
break;
}
}
for (int i=h[x];i;i=nx[i])
if (v[i]!=fa[x])
if (!mk[v[i]])
{
rc[x]=v[i];
break;
}
}
void dfs_2(int x)
{
if (lc[x]) dfs_2(lc[x]);
if (rc[x]) dfs_2(rc[x]);
while (!pq.empty() && fa[pq.top().x]==x)
st[++top]=pq.top(),pq.pop();
if (!pq.empty()) res=ma(res,pq.top().s+dep[x]);
while (top) pq.push(st[top--]);
pq.push((info){x,2*(mk[x]?0:dep[x]-dep[fa[x]])-dep[x]});
}
int main()
{
read(n);read(m);
for (int i=1;i<n;i++)
{
int uu,vv,ww;
read(uu);read(vv);read(ww);
egadd(uu,vv,ww);
egadd(vv,uu,ww);
}
dfs_1(1);
flag=check();
if (flag) dfs_3(1),dfs_2(1);
int x;
while (m--)
{
read(x);
if (flag) printf("%lld\n",dep[n]+mi(0,res+x));
else printf("%lld\n",dep[n]);
}
return 0;
}

CF1016F 【Road Projects】的更多相关文章

  1. 【26.83%】【Codeforces Round #380C】Road to Cinema

    time limit per test1 second memory limit per test256 megabytes inputstandard input outputstandard ou ...

  2. 【USACO 2017FEB】 Why Did the Cow Cross the Road III

    [题目链接] 点击打开链接 [算法] 树状数组 [代码] #include<bits/stdc++.h> using namespace std; #define MAXN 100010 ...

  3. 【USACO 2017Feb】 Why Did the Cow Cross the Road

    [题目链接] 点击打开链接 [算法] dist[i][j][k]表示当前走到(i,j),走的步数除以3的余数为k的最小花费 spfa即可 [代码] #include<bits/stdc++.h& ...

  4. 【POJ 3352】 Road Construction

    [题目链接] 点击打开链接 [算法] tarjan算法求边双联通分量 [代码] #include <algorithm> #include <bitset> #include ...

  5. 【Machine Learning】机器学习及其基础概念简介

    机器学习及其基础概念简介 作者:白宁超 2016年12月23日21:24:51 摘要:随着机器学习和深度学习的热潮,各种图书层出不穷.然而多数是基础理论知识介绍,缺乏实现的深入理解.本系列文章是作者结 ...

  6. 【微服务】SpringBoot、SpringCloud相关

    深入学习微框架:Spring Boot:   http://www.infoq.com/cn/articles/microframeworks1-spring-boot/ Spring Boot--2 ...

  7. 【Vegas原创】SVN的搭建及安装使用

    中文手册:http://tortoisesvn.net/docs/nightly/TortoiseSVN_zh_CN/index.html 所需程序: 1,TortoiseSVN  http://so ...

  8. 【BZOJ-3545&3551】Peaks&加强版 Kruskal重构树 + 主席树 + DFS序 + 倍增

    3545: [ONTAK2010]Peaks Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1202  Solved: 321[Submit][Sta ...

  9. 【C# -- OpenCV】Emgu CV 第一个实例

    原文 [C# -- OpenCV]Emgu CV 第一个实例 Emgu CV下载地址 http://sourceforge.net/projects/emgucv/files/ 找最新的下就行了,傻瓜 ...

随机推荐

  1. tp框架数据查询,增删改

    select()是数据模型的一个指定方法,可以获得数据表的数据信息 返回一个二维数组信息,当前数据表的全部数据信息 $obj = D();  创建对象 $obj -> select();  查询 ...

  2. sql-主键、外键、索引

    SQL的主键和外键的作用: 外键取值规则:空值或参照的主键值. (1)插入非空值时,如果主键表中没有这个值,则不能插入. (2)更新时,不能改为主键表中没有的值. (3)删除主键表记录时,你可以在建外 ...

  3. php ceil()函数 语法

    php ceil()函数 语法 ceil()函数怎么用? php ceil()函数的作用是向上舍入为最接近的整数,语法是ceil(number),表示返回不小于参数X的下一个整数,如果没有小数,返回参 ...

  4. 【dart学习】-- Dart之异步编程

    一,概述 编程中的代码执行,通常分为同步与异步两种. 同步:简单说,同步就是按照代码的编写顺序,从上到下依次执行,这也是最简单的我们最常接触的一种形式.但是同步代码的缺点也显而易见,如果其中某一行或几 ...

  5. uni-app获取元素宽高封装

    getElSize(id) { //得到元素的size return new Promise((res, rej) => { uni.createSelectorQuery().select(' ...

  6. 安装ISS服务

    二个操作系统 http://jingyan.baidu.com/article/5552ef471dcdd5518efbc976.html(win7)

  7. Sqli labs系列-less-5&6 报错注入法(上)

    在我一系列常规的测试后发现,第五关和第六关,是属于报错注入的关卡,两关的区别是一个是单引号一个是双引号...当然我是看了源码的.... 基于报错注入的方法,我早就忘的差不多了,,,我记的我最后一次基于 ...

  8. 爬虫问题之Unknown command: crawl

    出现这个问题,很大原因是爬虫没有在项目文件夹里运行,因为scrapy 这个爬虫框架封装好的一些命令,必须在框架内环境支持下才能运行 另外在环境目录下,还有很多命令,也必须在此路径环境下才能执行 可以通 ...

  9. Container 技能图谱skill-map

    # Container 技能图谱 ## 1. 容器核心 - [Docker](https://www.docker.com/) - [LXC](https://linuxcontainers.org/ ...

  10. C# Winform版批量压缩图片程序

    需求 上周,领导给我分配了一个需求:服务器上的图片文件非常大,每天要用掉两个G的大小的空间,要做一个自动压缩图片的工具处理这些大图片.领导的思路是这样的: 1)打开一个图片,看它的属性里面象素是多少, ...