淀粉质模板 Tree
2024-08-28 10:11:53
Tree
题目描述
给你一棵TREE,以及这棵树上边的距离.问有多少对点它们两者间的距离小于等于K
输入输出格式
输入格式:
N(n<=40000) 接下来n-1行边描述管道,按照题目中写的输入 接下来是k
输出格式:
一行,有多少对点之间的距离小于等于k
淀粉质感觉怎么写都不好看啊,迷。。
实现方法非常多。
大概思路:
对每一个子树的二层子节点进行遍历,处理每个点所属的二层子节点和到根节点的距离
以到根节点的距离为关键字排序,从两边进行扫描
如果当前满足,答案就加上\(r-l-\)和\(l\)属于同一颗二层子节点的数的数量,后者可以直接拿一个桶边扫描边维护
这个每次可以统计答案的区间是逐渐缩小的,有单调性。每次统计时候的意义是对\(l\)位置的节点,有多个点可以跨过根和它配对。
然后递归处理子树的答案。注意每次选择重心作为根节点保证复杂度。
Code:
#include <cstdio>
#include <algorithm>
const int N=4e4+10;
const int inf=0x3f3f3f3f;
int head[N],to[N<<1],Next[N<<1],edge[N<<1],cnt;
void add(int u,int v,int w)
{
to[++cnt]=v,Next[cnt]=head[u],edge[cnt]=w,head[u]=cnt;
}
struct node
{
int b,d;
node(){}
node(int b,int d){this->b=b,this->d=d;}
bool friend operator <(node n1,node n2){return n1.d<n2.d;}
}a[N];
int mi,id,ans,n,m,k,coun[N],siz[N],del[N];
int max(int x,int y){return x>y?x:y;}
void get_g(int now,int fa,int su)
{
siz[now]=1;int mx=0;
for(int i=head[now];i;i=Next[i])
{
int v=to[i];
if(!del[v]&&v!=fa)
{
get_g(v,now,su);
mx=max(mx,siz[v]);
siz[now]+=siz[v];
}
}
mx=max(mx,su-siz[now]);
if(mx<mi) mi=mx,id=now;
}
void dfs(int now,int fa,int anc,int dis)
{
a[++cnt]=node(anc,dis);
siz[now]=1;
for(int i=head[now];i;i=Next[i])
{
int v=to[i];
if(!del[v]&&v!=fa)
dfs(v,now,anc,dis+edge[i]),siz[now]+=siz[v];
}
}
void divide(int now,int su)
{
mi=inf,cnt=0;
get_g(now,0,su);
now=id;del[now]=1;
a[++cnt]=node(now,0),coun[now]=1;
for(int i=head[now];i;i=Next[i])
{
int v=to[i];
if(!del[v])
dfs(v,now,v,edge[i]),coun[v]=siz[v];
}
std::sort(a+1,a+1+cnt);
int l=1,r=cnt;
while(l<r)
{
while(l<r&&a[r].d+a[l].d>k) --coun[a[r--].b];
if(a[r].d+a[l].d<=k) ans+=r-l-coun[a[l].b]+1;
--coun[a[l++].b];
}
for(int i=head[now];i;i=Next[i])
{
int v=to[i];
if(!del[v])
divide(v,siz[v]);
}
}
int main()
{
scanf("%d",&n);
for(int u,v,w,i=1;i<n;i++)
{
scanf("%d%d%d",&u,&v,&w);
add(u,v,w),add(v,u,w);
}
scanf("%d",&k);
divide(1,n);
printf("%d\n",ans);
return 0;
}
2018.9.15
最新文章
- 为你的网页图标(Favicon)添加炫丽的动画和图片
- TNS-01251: Cannot set trace/log directory under ADR
- 结合Scikit-learn介绍几种常用的特征选择方法
- Swift - 文本输入框内容改变时响应,并获取最新内容
- Maven+Spring
- 简化布隆过滤器——BitMap
- PC版模块滚动不显示滚动条效果
- Redis 高可用集群
- Tomcat系列(10)——Tomcat主要设计模式5种(外观,责任链,观察者,模板方法,命令模式)
- WCF调错方法
- 为什么在Python里推荐使用多进程而不是多线程?(为什么python多线程无法增加CPU使用率?)
- docker lamp
- 使用go语言编写IOS和Android程序
- 生成ssh-key for GIthub
- 关于vue跨域名对接微信授权认证和APP授权认证
- 一、Linux概述 	二、Linux的安装 	三、Linux的常用命令(重点)
- element.style{}
- Pureftp SSL/TLS配置
- BZOJ4540 Hnoi2016 序列 【莫队+RMQ+单调栈预处理】*
- mongodb 副本集+分片集群搭建
热门文章
- P2661 信息传递 DFS
- docker swarm使用keepalived+haproxy搭建基于percona-xtradb-cluster方案的高可用mysql集群
- Mysql_Binary_Install_Scripts(采用二进制方式安装)
- 有一段<;script>;代码,效果是点击<;p>;就会弹出信息,但是有的<;p>;点击会有效果,有的没有效果
- Allowed memory size of 134217728 bytes exhausted (tried to allocate 2 bytes)
- php精华之独孤九剑
- sql语句(Oracle和sqlserver)
- [Bzoj4408]神秘数(主席树)
- Immutable
- Android的Fragment介绍