「HEOI2014」大工程
问题分析
首先不难想到是虚树。建完虚树需要保持节点间原先的距离关系。
然后总距离和最小距离用树形DP求,最大距离用两遍dfs即可。注意统计的时候只对关键点进行统计。
真是麻烦
参考程序
ac的时候是loj上速度最后一页,代码第四长的……
#include <bits/stdc++.h>
using namespace std;
const int Maxn = 1000010;
const long long INF = 1000000000010;
const int MaxLog = 20;
struct edge {
int To, Next;
long long Length;
edge() : To( 0 ), Next( 0 ), Length( 0LL ) {}
edge( int _To, int _Next, long long _Length ) :
To( _To ), Next( _Next ), Length( _Length ) {}
};
int n, q, k, A[ Maxn ], Important[ Maxn ];
int DFa[ Maxn ][ MaxLog ], Deep[ Maxn ], Dfn[ Maxn ], Time;
int Stack[ Maxn ];
int Flag[ Maxn ];
struct graph {
int Start[ Maxn ], Used, State;
edge Edge[ Maxn << 1 ];
graph() {}
inline void Init( int _DYT ) {
State = _DYT;
Used = 0;
return;
}
inline void AddDirectedEdge( int x, int y, long long Len ) {
if( Flag[ x ] != State ) {
Flag[ x ] = State;
Start[ x ] = 0;
}
Edge[ ++Used ] = edge( y, Start[ x ], Len );
Start[ x ] = Used;
return;
}
inline void AddUndirectedEdge( int x, int y, long long Len ) {
AddDirectedEdge( x, y, Len );
AddDirectedEdge( y, x, Len );
return;
}
};
graph Prime, Now;
long long Ans, Max, Min;
int Size[ Maxn ], Id;
void Build( int u, int Fa ) {
Deep[ u ] = Deep[ Fa ] + 1;
Dfn[ u ] = ++Time;
DFa[ u ][ 0 ] = Fa;
for( int i = 1; i < MaxLog; ++i )
DFa[ u ][ i ] = DFa[ DFa[ u ][ i - 1 ] ][ i - 1 ];
for( int t = Prime.Start[ u ]; t; t = Prime.Edge[ t ].Next ) {
int v = Prime.Edge[ t ].To;
if( v == Fa ) continue;
Build( v, u );
}
return;
}
inline bool Cmp( int x, int y ) {
return Dfn[ x ] < Dfn[ y ];
}
int GetLca( int x, int y ) {
if( Deep[ x ] < Deep[ y ] ) swap( x, y );
for( int i = MaxLog - 1; i >= 0; --i )
if( Deep[ DFa[ x ][ i ] ] >= Deep[ y ] )
x = DFa[ x ][ i ];
if( x == y ) return x;
for( int i = MaxLog - 1; i >= 0; --i )
if( DFa[ x ][ i ] != DFa[ y ][ i ] ) {
x = DFa[ x ][ i ];
y = DFa[ y ][ i ];
}
return DFa[ x ][ 0 ];
}
struct info {
long long Min, Sec;
info() : Min( INF ), Sec( INF ) {}
info( long long _Min, long long _Sec ) : Min( _Min ), Sec( _Sec ) {}
inline info operator + ( const long long Other ) const {
return info( Min + Other, Sec + Other );
}
inline info operator + ( const info Other ) const {
return ( Min < Other.Min ) ? info( Min, min( Sec, Other.Min ) ) : info( Other.Min, min( Min, Other.Sec ) ) ;
}
};
info GetMin( int u, int Fa ) {
info Ans = info( INF, INF );
if( Important[ u ] == Now.State ) Ans.Min = 0;
for( int t = Now.Start[ u ]; t; t = Now.Edge[ t ].Next ) {
int v = Now.Edge[ t ].To;
if( v == Fa ) continue;
Ans = Ans + ( GetMin( v, u ) + Now.Edge[ t ].Length );
}
Min = min( Min, Ans.Min + Ans.Sec );
return Ans;
}
void GetMax( int u, int Fa, long long Len ) {
if( Len > Max && Important[ u ] == Now.State ) {
Max = Len;
Id = u;
}
for( int t = Now.Start[ u ]; t; t = Now.Edge[ t ].Next ) {
int v = Now.Edge[ t ].To;
if( v == Fa ) continue;
GetMax( v, u, Len + Now.Edge[ t ].Length );
}
return;
}
long long GetAns( int u, int Fa ) {
Size[ u ] = 0; long long Sum = 0;
if( Important[ u ] == Now.State ) Size[ u ] = 1;
for( int t = Now.Start[ u ]; t; t = Now.Edge[ t ].Next ) {
int v = Now.Edge[ t ].To;
if( v == Fa ) continue;
long long SS = GetAns( v, u );
Ans += Sum * Size[ v ] + Size[ u ] * ( Now.Edge[ t ].Length * Size[ v ] + SS );
Sum += SS + Now.Edge[ t ].Length * Size[ v ];
Size[ u ] += Size[ v ];
}
return Sum;
}
void Work( int Case ) {
Now.Init( Case );
scanf( "%d", &k );
for( int i = 1; i <= k; ++i ) scanf( "%d", &A[ i ] );
for( int i = 1; i <= k; ++i ) Important[ A[ i ] ] = Case;
sort( A + 1, A + k + 1, Cmp );
Stack[ 0 ] = 1; Stack[ 1 ] = 1;
int Len, Lca;
for( int i = 1; i <= k; ++i ) {
if( i == 1 && A[ 1 ] == 1 ) continue;
if( i > 1 && A[ i ] == A[ i - 1 ] ) continue;
Lca = GetLca( Stack[ Stack[ 0 ] ], A[ i ] );
if( Deep[ Lca ] == Deep[ Stack[ Stack[ 0 ] ] ] )
Stack[ ++Stack[ 0 ] ] = A[ i ];
else {
while( Deep[ Stack[ Stack[ 0 ] - 1 ] ] > Deep[ Lca ] ) {
Len = Deep[ Stack[ Stack[ 0 ] ] ] - Deep[ Stack[ Stack[ 0 ] - 1 ] ];
Now.AddUndirectedEdge( Stack[ Stack[ 0 ] - 1 ], Stack[ Stack[ 0 ] ], Len );
--Stack[ 0 ];
}
if( Deep[ Stack[ Stack[ 0 ] - 1 ] ] == Deep[ Lca ] ) {
Len = Deep[ Stack[ Stack[ 0 ] ] ] - Deep[ Stack[ Stack[ 0 ] - 1 ] ];
Now.AddUndirectedEdge( Stack[ Stack[ 0 ] - 1 ], Stack[ Stack[ 0 ] ], Len );
--Stack[ 0 ];
Stack[ ++Stack[ 0 ] ] = A[ i ];
} else {
Len = Deep[ Stack[ Stack[ 0 ] ] ] - Deep[ Lca ];
Now.AddUndirectedEdge( Stack[ Stack[ 0 ] ], Lca, Len );
--Stack[ 0 ];
Stack[ ++Stack[ 0 ] ] = Lca;
Stack[ ++Stack[ 0 ] ] = A[ i ];
}
}
}
while( Stack[ 0 ] > 1 ) {
Len = Deep[ Stack[ Stack[ 0 ] ] ] - Deep[ Stack[ Stack[ 0 ] - 1 ] ];
Now.AddUndirectedEdge( Stack[ Stack[ 0 ] ], Stack[ Stack[ 0 ] - 1 ], Len );
--Stack[ 0 ];
}
Min = INF;
GetMin( 1, 0 );
Max = -1;
GetMax( A[ 1 ], 0, 0 );
Max = -1;
GetMax( Id, 0, 0 );
Ans = 0;
GetAns( 1, 0 );
printf( "%lld %lld %lld\n", Ans, Min, Max );
return;
}
int main() {
scanf( "%d", &n );
for( int i = 1; i < n; ++i ) {
int x, y;
scanf( "%d%d", &x, &y );
Prime.AddUndirectedEdge( x, y, 1 );
}
Build( 1, 0 );
scanf( "%d", &q );
for( int i = 1; i <= q; ++i ) Work( i );
return 0;
}
「HEOI2014」大工程的更多相关文章
- 「ZJOI2016」大森林 解题报告
「ZJOI2016」大森林 神仙题... 很显然线段树搞不了 考虑离线操作 我们只搞一颗树,从位置1一直往后移动,然后维护它的形态试试 显然操作0,1都可以拆成差分的形式,就是加入和删除 因为保证了操 ...
- LOJ#2230. 「BJOI2014」大融合
LOJ#2230. 「BJOI2014」大融合 题目描述 小强要在$N$个孤立的星球上建立起一套通信系统.这套通信系统就是连接$N$个点的一个树.这个树的边是一条一条添加上去的. 在某个时刻,一条边的 ...
- 【HEOI2014】大工程<虚树>
虚树 我们每天都用心思索着,这究竟是为了什么呢?我想我也不知道,只是觉得如果人不思考问题就很无聊. 我觉得虚树不是什么数据结构,就是一种技巧或者工具.它能把树中\(k\)个关键点以\(O(klogk) ...
- 【BZOJ】【3611】【HEOI2014】大工程
虚树+树形DP 本题100W的点数……不用虚树真的好吗…… Orz ZYF 我的感悟: dp的过程跟SPOJ 1825 FTOUR2 的做法类似,依次枚举每个子树,从当前子树和之前的部分中各找一条最长 ...
- 「HEOI2014」南园满地堆轻絮
题目链接 戳我 题目出处 菩萨蛮·南园满地堆轻絮 温庭筠 南园满地堆轻絮,愁闻一霎清明雨.雨后却斜阳,杏花零落香 ...
- @loj - 2092@ 「ZJOI2016」大森林
目录 @description@ @solution@ @accepted code@ @details@ @description@ 小 Y 家里有一个大森林,里面有 n 棵树,编号从 1 到 n. ...
- 【LOJ】#2230. 「BJOI2014」大融合
题解 我现在真是太特么老年了 一写数据结构就颓废,难受 这题就是用lct维护子树 ???lct怎么维护子树 这样想,我们给每个点记录虚边所在的子树大小,只发生在Access和link的时候 这样的话我 ...
- loj2230 「BJOI2014」大融合
LCT裸题 我LCT学傻了这题明显可以树剖我不会树剖了 本来的siz是Splay上的子树和,并没有什么用. 所以每个点维护虚子树和和子树和 虚子树和即虚边连接的子树和,且只有在access和link操 ...
- loj2092 「ZJOI2016」大森林
ref不是太懂-- #include <algorithm> #include <iostream> #include <cstring> #include < ...
随机推荐
- web框架链接
django系列教程(优):https://www.cnblogs.com/feixuelove1009/p/5823135.html
- mybatis中collection子查询注入参数为null
具体实现参照网上,但是可能遇到注入参数为null的情况,经过查阅及自己测试记录一下: 子查询的参数中,有<if test="">之类,需要指定别名,通过 http:// ...
- pycharm2017.3.3永久激活(转载)
pycharm是很强大的开发工具,但是每次注册着实让人头疼.网络上很多注册码.注册服务器等等.但都只是一年或者不能用:为次有如下解决方案.亲测有效!!! 如果想让pycharm永久被激活,比如截止日到 ...
- [转载]python with语句的用法
https://www.cnblogs.com/DswCnblog/p/6126588.html 看这篇文章的时候看到了python的类名()用法,很好奇,上网查了下,原来这就相当于对类进行实例化了. ...
- 能当壁纸用的Git常用命令速查表
使用Microsoft Office 2016手工绘制. 链接: https://pan.baidu.com/s/18KsH-u5T2iSTHaXd6iQWGA 提取码: w8da 复制这段内容后打开 ...
- 记录--js中出现的数组排序问题
这是今天在写vue项目时发生的一个小问题,在此记录一下,方便自己的回顾.项目是前后端分离的,前台主要使用了vue-cli3.0 + mintui,是一个移动端的web app包括了后台发布管理的一些功 ...
- pandas中Series对象下的str所拥有的方法(df["xx"].str)
在使用pandas的时候,经常要对DataFrame的某一列进行操作,一般都会使用df["xx"].str下的方法,但是都有哪些方法呢?我们下面来罗列并演示一下.既然是df[&qu ...
- SQL语句复习【专题一】
SQL语句复习[专题一] --创建用户 scott 并设置密码为 tiger create user scott identified by tiger --用户刚刚创建没有任何的权限,连登录的权限都 ...
- maven报错 java.lang.RuntimeException: com.google.inject.CreationException: Unable to create injector, see the following errors
2 errors java.lang.RuntimeException: com.google.inject.CreationException: Unable to create injector, ...
- 给移动硬盘装win10,知道这些就足够了
随着制造工业的不断发展,储存介质逐渐廉价化,以某猫和某狗为代表的电商平台上都能轻松买到大容量的原装移动硬盘.如果工厂的产品还不能满足你的要求,那么DIY是一个不错的选择,可以选择购买移动硬盘盒(2.5 ...