BZOJ 3365 Distance Statistics 点分治

时间:2021-07-20 23:35:45

这道题是一道点分治的题目,难度不大,可以拿来练手。

关键是对于找出来的重心的删除操作需要删掉这条边,这很重要。

还有每次找重心的时候,不但要考虑他的子节点的siz,还要考虑父节点的siz。

然后就A了。。。

每次点分治 分两种情况讨论一下就可以啦!

BZOJ 3365 Distance Statistics 点分治

/w\...

#include <cstdio>
#include <algorithm> using namespace std; const int maxn = 56789;
const int INF = 1000000007; int getint()
{
int r = 0, k = 1;
char c;
for (c = getchar(); c < '0' || c > '9'; c = getchar() ) if (c == '-') k = -1;
for (; '0' <= c && c <= '9'; c = getchar() ) r = r * 10 - '0' + c;
return r * k;
} int n, m, k;
int h[maxn], siz[maxn]; struct edge_type
{
int v, next, w;
bool baned;
} edge[maxn * 2]; int tote = 0; void ins(int u, int v, int w)
{
edge[++tote].v = v;
edge[tote].next = h[u];
edge[tote].w = w;
edge[tote].baned = false;
h[u] = tote;
} int fgans, fgsiz; void get_fgans(int now, int fa, int size) {
int tmp = 1;
siz[now] = 1;
for (int i = h[now]; i; i = edge[i].next) {
if (!edge[i].baned && fa != edge[i].v) {
get_fgans(edge[i].v, now, size);
siz[now] += siz[edge[i].v];
if (tmp < siz[edge[i].v]) tmp = siz[edge[i].v];
}
}
if (tmp < size - siz[now]) tmp = size - siz[now];
if (tmp < fgsiz) {
fgans = now;
fgsiz = tmp;
}
}
int find_gravity(int rt, int size) {
fgsiz = INF;
fgans = rt;
get_fgans(rt, -1, size);
return fgans;
} int dis[maxn];
int zhan[maxn], zcnt; void get_dis(int now, int nd, int fa) {
dis[now] = nd;
for (int i = h[now]; i; i = edge[i].next)
if (!edge[i].baned && fa != edge[i].v)
get_dis(edge[i].v, nd + edge[i].w, now);
} void dfs(int now, int fa) {
zhan[++zcnt] = dis[now];
for (int i = h[now]; i; i = edge[i].next)
if (!edge[i].baned && fa != edge[i].v)
dfs(edge[i].v, now);
} int calculate(int rt) {
int ret = 0;
zcnt = 0;
dfs(rt, -1);
sort(zhan + 1, zhan + zcnt + 1);
for (int i = 1, j = zcnt; i <= zcnt; ++i) {
for (; j && zhan[i] + zhan[j] > k; --j);
ret += j;
}
return ret;
} int Ans = 0; void dfz(int now, int size)
{
int rt = find_gravity(now, size);
get_dis(rt, 0, -1);
Ans += calculate(rt);
int v;
for (int i = h[rt]; i; i = edge[i].next)
if (!edge[i].baned) {
edge[i].baned = edge[((i-1)^1)+1].baned = true;
Ans -= calculate(edge[i].v);
dfz(edge[i].v, siz[edge[i].v]);
}
} int main()
{
n = getint();
m = getint();
int u, v, w;
for (int i = 1; i < n; ++i)
{
u = getint();
v = getint();
w = getint();
ins (u, v, w);
ins (v, u, w);
}
k = getint();
dfz(1, n);
printf("%d", (Ans - n >> 1));
return 0;
}

BZOJ 3365 Distance Statistics 点分治的更多相关文章

  1. POJ 1987 BZOJ 3365 Distance Statistics 树的分治&lpar;点分治&rpar;

    题目大意:(同poj1741,刷一赠一系列) CODE: #include <cstdio> #include <cstring> #include <iostream& ...

  2. &lbrack;BZOJ 3365&rsqb; Distance Statistics

    [题目链接] https://www.lydsy.com/JudgeOnline/problem.php?id=3365 [算法] 点分治 [代码] #include <algorithm&gt ...

  3. POJ 1987 Distance Statistics 树分治

    Distance Statistics     Description Frustrated at the number of distance queries required to find a ...

  4. BZOJ&lowbar;3365&lowbar;&lbrack;Usaco2004 Feb&rsqb;Distance Statistics 路程统计&amp&semi;&amp&semi;POJ&lowbar;1741&lowbar;Tree&lowbar;点分治

    BZOJ_3365_[Usaco2004 Feb]Distance Statistics 路程统计&&POJ_1741_Tree_点分治 Description     在得知了自己农 ...

  5. BZOJ&period;4184&period;shallot&lpar;线段树分治 线性基&rpar;

    BZOJ 裸的线段树分治+线性基,就是跑的巨慢_(:з」∠)_ . 不知道他们都写的什么=-= //41652kb 11920ms #include <map> #include < ...

  6. &lbrack;BZOJ 4025&rsqb;二分图&lpar;线段树分治&plus;带边权并查集&rpar;

    [BZOJ 4025]二分图(线段树分治+带边权并查集) 题面 给出一个n个点m条边的图,每条边会在时间s到t出现,问每个时间的图是否为一个二分图 \(n,m,\max(t_i) \leq 10^5\ ...

  7. bzoj 3365 &lbrack;Usaco2004 Feb&rsqb;Distance Statistics 路程统计(点分治,单调)

    [题意] 求树上长度不超过k的点对数目. [思路] 和 Tree 一样一样的. 就是最后统计的时候别忘把根加上. [代码] #include<set> #include<cmath& ...

  8. bzoj 3365&colon; &lbrack;Usaco2004 Feb&rsqb;Distance Statistics 路程统计【容斥原理&plus;点分治】

    统计在一个root下的两个子树,每个子树都和前面的运算一下再加进去对于这种需要排序的运算很麻烦,所以考虑先不去同子树内点对的算出合法点对个数,然后减去每一棵子树内的合法点对(它们实际上是不合法的,相当 ...

  9. BZOJ 3365&colon; &lbrack;Usaco2004 Feb&rsqb;Distance Statistics 路程统计

    Description 一棵树,统计距离不大于 \(k\) 的点对个数. Sol 点分治. 发现自己快把点分治忘干净了... 找重心使所有儿子的最大值尽量小,然后每次处理全部子树,再减去每个子树的贡献 ...

随机推荐

  1. linux-dns服务器搭建

    1.先查看系统是否安装了bind rpm -qa|grep bind 2.如果没有安装则

  2. 为Unity项目生成文档&lpar;二&rpar;

    Unity项目生成文档 接着上篇文章:为Unity项目生成文档(一) .Net项目可在VS配置XML 我们可以在VS中通过配置来生成xml文件,但是unity的project,就算同样配置了xml文档 ...

  3. html5自带表单验证-美化改造

    神奇的代码 暂且叫做html5.css /* === HTML5 validation styles === */ .myform select:required, .myform input:req ...

  4. 《Genesis-3D开源游戏引擎--横版格斗游戏制作教程02:关键帧动画导入与切割》

    2. 关键帧动画导入与切割 动画的分割与导入概述: 在游戏当中,游戏角色在不同状态下会有不同的动作,这些动作在引擎里相当于一段段的动画片段.当导入模型资源的时候,连同模型动画都会一并导入到引擎中.开发 ...

  5. Linux中的随机数文件 &sol;dev&sol;random &sol;dev&sol;urandom

    Linux中的随机数可以从两个特殊的文件中产生,一个是/dev/urandom.另外一个是/dev/random.他们产生随机数的原理是利用当前系统的熵池来计算出固定一定数量的随机比特,然后将这些比特 ...

  6. easy ui tree 取父节点的值和取蓝色框的值

    var nodes = $('#basetree').tree('getChecked'); var cnode = ''; var fnode = ''; for ( var i = 0; i &l ...

  7. Python之路&comma;Day25-----暂无正在更新中

    Python之路,Day25-----暂无正在更新中

  8. HDU OJ 4334 Trouble 2012 Multi-University Training Contest 4

    题目:click here 题意: 给定5组数据,每组数据选择一个数,看是否能找到5个数的和为零. 分析: 千万不要~~T~~ 普通线性查找: #include <iostream> #i ...

  9. 跟着大神重写的KNN 文档归类小工具

    ·背景 在知道KNN之前,楼主有时候会粗糙地做一些分类模型的计算.在拜读了Orisun大神[http://www.cnblogs.com/zhangchaoyang/articles/2162393. ...

  10. linux下安装部署环境:jdk、tomcat、nginx

    一.安装jdk 一.查看Linux自带的JDK是否已安装 1.查看已经安装的jdk: # rpm -qa|grep jdk                ← 查看jdk的信息或直接执行 或 # rpm ...