HDU 5016 Mart Master II (树上点分治)
题目地址: pid=5016">HDU 5016
先两遍DFS预处理出每一个点距近期的基站的距离与基站的编号。
然后找重心。求出每一个点距重心的距离。然后依据dis[x]+dis[y] < d[y]。用二分找出当前子树中不会被占据的数量,总点数减去即是被占据的数量。
这样就能够求出每一个点最多占据的点的数量。然后找最大值就可以。
代码例如以下:
#include <iostream>
#include <string.h>
#include <math.h>
#include <queue>
#include <algorithm>
#include <stdlib.h>
#include <map>
#include <set>
#include <stdio.h>
#include <time.h>
using namespace std;
#define LL __int64
#define pi acos(-1.0)
#pragma comment(linker, "/STACK:1024000000")
const int mod=1e9+7;
const int INF=0x3f3f3f3f;
const double eqs=1e-9;
const int MAXN=100000+10;
int head[MAXN], cnt, root, min1, tot;
int siz[MAXN], vis[MAXN], d[MAXN], id[MAXN], ans[MAXN], dis[MAXN];
int color[MAXN];
struct node
{
int v, w, next;
}edge[MAXN<<1];
void add(int u, int v, int w)
{
edge[cnt].v=v;
edge[cnt].w=w;
edge[cnt].next=head[u];
head[u]=cnt++;
}
struct N
{
int dis, id;
}F[MAXN];
bool cmp(N x, N y)
{
return x.dis<y.dis||(x.dis==y.dis&&x.id<y.id);
}
void init()
{
memset(head,-1,sizeof(head));
memset(vis,0,sizeof(vis));
memset(d,INF,sizeof(d));
memset(id,0,sizeof(id));
memset(ans,0,sizeof(ans));
cnt=0;
}
void dfs1(int u, int fa)
{
if(color[u]){
id[u]=u;
d[u]=0;
}
for(int i=head[u];i!=-1;i=edge[i].next){
int v=edge[i].v;
if(v==fa) continue ;
dfs1(v,u);
if(d[u]>d[v]+edge[i].w){
d[u]=d[v]+edge[i].w;
id[u]=id[v];
}
else if(d[u]==d[v]+edge[i].w)
id[u]=min(id[u],id[v]);
}
}
void dfs2(int u, int fa)
{
for(int i=head[u];i!=-1;i=edge[i].next){
int v=edge[i].v;
if(v==fa) continue ;
if(d[v]>d[u]+edge[i].w){
d[v]=d[u]+edge[i].w;
id[v]=id[u];
}
else if(d[v]==d[u]+edge[i].w){
id[v]=min(id[v],id[u]);
}
dfs2(v,u);
}
}
void getsize(int u, int fa)
{
siz[u]=1;
for(int i=head[u];i!=-1;i=edge[i].next){
int v=edge[i].v;
if(v==fa||vis[v]) continue ;
getsize(v,u);
siz[u]+=siz[v];
}
}
void getroot(int u, int fa, int s)
{
int max1=0;
for(int i=head[u];i!=-1;i=edge[i].next){
int v=edge[i].v;
if(v==fa||vis[v]) continue ;
getroot(v,u,s);
max1=max(max1,siz[v]);
}
max1=max(max1,s-siz[u]);
if(min1>max1){
min1=max1;
root=u;
}
}
void getdis(int u, int fa, int l)
{
dis[u]=l;
F[tot].dis=d[u]-l;
F[tot++].id=id[u];
for(int i=head[u];i!=-1;i=edge[i].next){
int v=edge[i].v;
if(v==fa||vis[v]) continue ;
getdis(v,u,l+edge[i].w);
}
}
int BS(int l, int u)
{
int low=0, high=tot-1, mid, ans=-1;
while(low<=high){
mid=low+high>>1;
if(F[mid].dis<l||(F[mid].dis==l&&F[mid].id<=u)){
low=mid+1;
ans=mid;
}
else high=mid-1;
}
return ans+1;
}
void getans(int u, int fa, int f, int s)
{
int tmp=BS(dis[u],u);
ans[u]+=f*(s-tmp);
for(int i=head[u];i!=-1;i=edge[i].next){
int v=edge[i].v;
if(v==fa||vis[v]) continue ;
getans(v,u,f,s);
}
}
void Cal(int u, int l, int f)
{
tot=0;
getdis(u,-1,l);
sort(F,F+tot,cmp);
getans(u,-1,f,siz[u]);
}
void work(int u)
{
vis[u]=1;
getsize(u,-1);
Cal(u,0,1);
for(int i=head[u];i!=-1;i=edge[i].next){
int v=edge[i].v;
if(vis[v]) continue ;
Cal(v,edge[i].w,-1);
min1=INF;
getroot(v,u,siz[v]);
work(root);
}
}
int main()
{
int n, u, v, w, i, res;
while(scanf("%d",&n)!=EOF){
init();
for(i=1;i<n;i++){
scanf("%d%d%d",&u,&v,&w);
add(u,v,w);
add(v,u,w);
}
for(i=1;i<=n;i++){
scanf("%d",&color[i]);
}
dfs1(1,-1);
dfs2(1,-1);
min1=INF;
getsize(1,-1);
getroot(1,-1,n);
work(root);
res=0;
for(i=1;i<=n;i++){
if(color[i]) continue ;
res=max(res,ans[i]);
}
printf("%d\n",res);
}
return 0;
}
HDU 5016 Mart Master II (树上点分治)的更多相关文章
- HDU 5016 Mart Master II
Mart Master II Time Limit: 6000ms Memory Limit: 65536KB This problem will be judged on HDU. Original ...
- 【点分治】hdu5016 Mart Master II
点分治好题. ①手动开栈. ②dp预处理每个点被哪个市场控制,及其距离是多少,记作pair<int,int>数组p. ③设dis[u].first为u到重心s的距离,dis[u].seco ...
- HDU - 6268: Master of Subgraph (分治+bitset优化背包)
题意:T组样例,给次给出一个N节点的点权树,以及M,问连通块的点权和sum的情况,输出sum=1到M,用0或者1表示. 思路:背包,N^2,由于是无向的连通块,所以可以用分治优化到NlgN. 然后背包 ...
- hdu 5016 点分治(2014 ACM/ICPC Asia Regional Xi'an Online)
Mart Master II Time Limit: 12000/6000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others)T ...
- codeforces 161D Distance in Tree 树上点分治
链接:https://codeforces.com/contest/161/problem/D 题意:给一个树,求距离恰好为$k$的点对是多少 题解:对于一个树,距离为$k$的点对要么经过根节点,要么 ...
- POJ 1741 Tree 树上点分治
题目链接:http://poj.org/problem?id=1741 题意: 给定一棵包含$n$个点的带边权树,求距离小于等于K的点对数量 题解: 显然,枚举所有点的子树可以获得答案,但是朴素发$O ...
- HDU 3081 Marriage Match II(二分法+最大流量)
HDU 3081 Marriage Match II pid=3081" target="_blank" style="">题目链接 题意:n个 ...
- HDU 3081 Marriage Match II (网络流,最大流,二分,并查集)
HDU 3081 Marriage Match II (网络流,最大流,二分,并查集) Description Presumably, you all have known the question ...
- HDU 3081 Marriage Match II (二分图,并查集)
HDU 3081 Marriage Match II (二分图,并查集) Description Presumably, you all have known the question of stab ...
随机推荐
- Akka源码分析-Router
akak中还有一个比较重要的概念,那就是Router(路由).路由的概念,相信大家都不陌生,在akka中,它就是其他actors的一个代理,会把消息按照路由规则,分发给指定的actor.我一般喜欢把R ...
- 2 我们的C#学习方法
在这里我们借鉴了一种行之有效的学习编程语言的方法,并在此基础上进行了相应的改良.我们在培训新人中使用后,发现这种学习方法是非常有效的. 你通过做以下几个步骤来一步步学习C#语言. 1. 搞懂每一个我们 ...
- MVC系列学习(十四)-路由规则及路由调试工具
1.本次学习的代码,比较简单,就是在路由配置文件中,添加一个路由信息:同时添加一个相应的控制器及视图 控制器中代码如下 即有两条路由匹配规则,一个Kim控制器,该控制器下有个Index的方法,和一个对 ...
- CSS——宠物demo
注意:ul中自带padding值,需要清除. <!DOCTYPE html> <html lang="en"> <head> <meta ...
- GridView动态计算高度
// 动态加载GridView 高度 public static void setListViewHeightBasedOnChildren(MyGridView myGridView) { List ...
- 离线安装Selenium
https://blog.csdn.net/poem_ruru/article/details/79032140
- 汇总——WEB前端资源网
前端攻城师 爱思资源网 HTML5吧 0101后花园 前端网 编程教程和源代码示例 Javascript中文网 Web前端资源网 移动端HTML5资源整理 Web开发者 SegmentFault 前端 ...
- (转)Hibernate中的多表操作
http://blog.csdn.net/yerenyuan_pku/article/details/70556208 Hibernate中的多表操作 在实际开发中,我们不可能只是简简单单地去操作单表 ...
- 彩色MT9V034摄像头 Bayer转rgb FPGA实现
1 图像bayer格式介绍 bayer格式是伊士曼·柯达公司科学家Bryce Bayer发明的,Bryce Bayer所发明的拜耳阵列被广泛运用数字图像.Bayer格式是相机内部的原始数据, 一般后缀 ...
- function&箭头函数
JS中this到底指向谁? function:谁调用指向谁 var id = '654321' var handler = { id: '123456', init: function() { con ...