Acwing252.树

题目分析

树中的路径分为三种

  • 路径两端在同一个子树
  • 路径两端在不同子树
  • 路径有一端是重心

因此可以分情况处理, 对于第一种情况可以进行递归处理, 第二种情况需要使用容斥原理求得(下面重点介绍), 第三种情况枚举重心到其他节点的路径就可以求得

代码分析

求子树大小

int get_size(int _index, int pre) {
// 如果当前子树被删除, 返回0
if (visited[_index]) return 0; int res = 1;
for (int i = head[_index]; ~i; i = next_edge[i]) {
int ver = edge_end[i];
if (ver == pre) continue;
res += get_size(ver, _index);
} return res;
}

求当前点出发的子树的所有路径

void get_dist(int _index, int pre, int dis, int &ptr) {
if (visited[_index]) return;
arr2[ptr++] = dis;
for (int i = head[_index]; ~i; i = next_edge[i]) {
int ver = edge_end[i];
if (ver == pre) continue;
get_dist(ver, _index, dis + w[i], ptr);
}
}

求树的重心

int get_weight_center(int _index, int pre, int tot, int &res) {
if (visited[_index]) return 0;
int sum = 1, max_son = 0; for (int i = head[_index]; ~i; i = next_edge[i]) {
int ver = edge_end[i];
if (ver == pre) continue;
int size = get_weight_center(ver, _index, tot, res);
// 更新最大子树大小
max_son = max(max_son, size);
sum += size;
} // 总数减下面的子树总和, 就是当前点所在父节点的子树大小
max_son = max(max_son, tot - sum);
// 如果最大子树大小 <= tot / 2, 那么当前点可以作为重心
if (max_son <= tot >> 1) res = _index;
return sum;
}

注意:这里树的重心只要子树的大小<= n / 2就可以保证点分治时间复杂度

求单个子树中路径和 <= k的路径组合数量

int get(int arr[], int cnt) {
sort(arr, arr + cnt);
int res = 0; for (int i = cnt - 1, j = -1; i >= 0; --i) {
// 找到满足条件的最大j
while (j + 1 < i && arr[j + 1] + arr[i] <= k) j++;
j = min(j, i - 1);
res += j + 1;
} return res;
}

双指针算法求解

核心函数

int calc(int _index) {
if (visited[_index]) return 0;
int res = 0; // 找到当前子树的重心
get_weight_center(_index, -1, get_size(_index, -1), _index);
visited[_index] = true; int p1 = 0;
for (int i = head[_index]; ~i; i = next_edge[i]) {
int ver = edge_end[i], p2 = 0;
get_dist(ver, -1, w[i], p2);
// 减去一个子树中的组合情况
res -= get(arr2, p2); // 将子树的距离加入到去全局的距离中
for (int j = 0; j < p2; ++j) {
// 某个点是重心的情况
if (arr2[j] <= k) res++;
arr1[p1++] = arr2[j];
}
} // 加上子树内部的组合情况
res += get(arr1, p1);
// 递归处理子树
for (int i = head[_index]; ~i; i = next_edge[i]) {
int ver = edge_end[i];
res += calc(ver);
}
return res;
}

先找到当前子树的重心, 然后将重心标记

然后将路径分为三种情况, 上面有论述,

直接看最复杂的情况, 也就是两个点分别在两个子树中

  • 求得子树中的路径组合, 因为是不合法的, 因此将其从答案中减去
  • 然后将子树的路径添加到全局的路径当中
  • 在添加过程中不要忘记记录子树到重心也是一种情况
  • 最后加入到res的就是全局的除去子树之间的路径组合

最后处理第一种和第三种情况

时间复杂度分析

递归过程中一共$\log{}{n}$层, 每一层点数不超过$n$, 排序时间复杂度$O(n\log_{}{n})$, 因此总的时间复杂度$O(n\log_{}{n}^{2})$

完整代码

#include <iostream>
#include <algorithm>
#include <cstring> using namespace std; const int N = 10010, EDGE_NUMBER = N << 1; int n, k;
int head[N], edge_end[EDGE_NUMBER], next_edge[EDGE_NUMBER], w[EDGE_NUMBER], edge_index;
//节点是否被删除
bool visited[N];
//当前重心的所有子树的距离, 当前子树的距离
int arr1[N], arr2[N]; void add(int ver1, int ver2, int val) {
edge_end[edge_index] = ver2, next_edge[edge_index] = head[ver1], w[edge_index] = val, head[ver1] = edge_index++;
} //求子树大小
int get_size(int _index, int pre) {
// 如果当前子树被删除, 返回0
if (visited[_index]) return 0; int res = 1;
for (int i = head[_index]; ~i; i = next_edge[i]) {
int ver = edge_end[i];
if (ver == pre) continue;
res += get_size(ver, _index);
} return res;
} //求从当前点出发所有路径的长度
void get_dist(int _index, int pre, int dis, int &ptr) {
if (visited[_index]) return;
arr2[ptr++] = dis;
for (int i = head[_index]; ~i; i = next_edge[i]) {
int ver = edge_end[i];
if (ver == pre) continue;
get_dist(ver, _index, dis + w[i], ptr);
}
} //求树的重心, 返回值是当前子树大小
int get_weight_center(int _index, int pre, int tot, int &res) {
if (visited[_index]) return 0;
int sum = 1, max_son = 0; for (int i = head[_index]; ~i; i = next_edge[i]) {
int ver = edge_end[i];
if (ver == pre) continue;
int size = get_weight_center(ver, _index, tot, res);
// 更新最大子树大小
max_son = max(max_son, size);
sum += size;
} // 总数减下面的子树总和, 就是当前点所在父节点的子树大小
max_son = max(max_son, tot - sum);
// 如果最大子树大小 <= tot / 2, 那么当前点可以作为重心
if (max_son <= tot >> 1) res = _index;
return sum;
} int get(int arr[], int cnt) {
sort(arr, arr + cnt);
int res = 0; for (int i = cnt - 1, j = -1; i >= 0; --i) {
// 找到满足条件的最大j
while (j + 1 < i && arr[j + 1] + arr[i] <= k) j++;
j = min(j, i - 1);
res += j + 1;
} return res;
} int calc(int _index) {
if (visited[_index]) return 0;
int res = 0; // 找到当前子树的重心
get_weight_center(_index, -1, get_size(_index, -1), _index);
visited[_index] = true; int p1 = 0;
for (int i = head[_index]; ~i; i = next_edge[i]) {
int ver = edge_end[i], p2 = 0;
get_dist(ver, -1, w[i], p2);
// 减去一个子树中的组合情况
res -= get(arr2, p2); // 将子树的距离加入到去全局的距离中
for (int j = 0; j < p2; ++j) {
// 某个点是重心的情况
if (arr2[j] <= k) res++;
arr1[p1++] = arr2[j];
}
} // 加上子树内部的组合情况
res += get(arr1, p1);
// 递归处理子树
for (int i = head[_index]; ~i; i = next_edge[i]) {
int ver = edge_end[i];
res += calc(ver);
}
return res;
} int main() {
while (scanf("%d%d", &n, &k), n && k) {
memset(visited, false, sizeof visited);
memset(head, -1, sizeof head);
edge_index = 0; for (int i = 0; i < n - 1; ++i) {
int ver1, ver2, val;
scanf("%d%d%d", &ver1, &ver2, &val);
add(ver1, ver2, val);
add(ver2, ver1, val);
} printf("%d\n", calc(0));
}
return 0;
}

男人八题-点分治-Acwing252.树的更多相关文章

  1. poj 1741 楼教主男人八题之中的一个:树分治

    http://poj.org/problem? id=1741 Description Give a tree with n vertices,each edge has a length(posit ...

  2. Cogs 1714. [POJ1741][男人八题]树上的点对(点分治)

    [POJ1741][男人八题]树上的点对 ★★★ 输入文件:poj1741_tree.in 输出文件:poj1741_tree.out 简单对比 时间限制:1 s 内存限制:256 MB [题目描述] ...

  3. poj 1737男人八题之一 orz ltc

    这是楼教主的男人八题之一.很高兴我能做八分之一的男人了. 题目大意:求有n个顶点的连通图有多少个. 解法: 1.  用总数减去不联通的图(网上说可以,我觉得时间悬) 2.    用动态规划(数学递推) ...

  4. POJ1742 Coins(男人八题之一)

    前言 大名鼎鼎的男人八题,终于见识了... 题面 http://poj.org/problem?id=1742 分析 § 1 多重背包 这很显然是一个完全背包问题,考虑转移方程: DP[i][j]表示 ...

  5. 新男人八题---AStringGame

    终于完成进度男人1/8,为了这题学了sam= = 题意先有一个串,n个子串,两个人轮流每次在子串上加字符,要求加完后还是原串的子串,最后不能加的就是输者,求赢的人 解法:sam之后在构造的状态图上跑s ...

  6. poj 1742(好题,楼天城男人八题,混合背包)

    Coins Time Limit: 3000MS   Memory Limit: 30000K Total Submissions: 33269   Accepted: 11295 Descripti ...

  7. 博弈论(男人八题):POJ 1740 A New Stone Game

    A New Stone Game Time Limit: 1000MS   Memory Limit: 30000K Total Submissions: 5694   Accepted: 3119 ...

  8. poj 1743 男人八题之后缀数组求最长不可重叠最长重复子串

    Musical Theme Time Limit: 1000MS   Memory Limit: 30000K Total Submissions: 14874   Accepted: 5118 De ...

  9. nyoj137 取石子(三) 楼教主男人八题之一

    思路:一堆时,N态.两堆时,当两堆数量相同,P态,不同为N态.三堆时,先手可以变成两堆一样的,必胜N态. 此时可以总结规律:堆数为偶数可能且石子数都是两两相同的,为P态.分析四堆时,当四堆中两两数量一 ...

  10. 洛谷T44252 线索_分治线段树_思维题

    分治线段树,其实就是将标记永久化,到最后再统一下传所有标记. 至于先后顺序,可以给每个节点开一个时间戳. 一般地,分治线段树用于离线,只查询一次答案的题目. 本题中,标记要被下传 222 次. Cod ...

随机推荐

  1. git针对指定网站设置代理

    我们经常要用到各种git地址,比如github.gitee还有自己搭建的git等等. 但是github我们经常拉取和推送代码的时候超时,这时候如果我们搜索会发现大量的文章都是告诉我们设置全局系统代理: ...

  2. Element-Plus表格:Table自定义合并行数据的最佳实践

    " 知行合一 " -- 王阳明 在开发项目中,我们时常会用到表格,许多需求可能会要求自定义特定的行或列. 接下来,我们将探讨在实际开发中如何应对这一挑战. 本文案例采用的技术: 名 ...

  3. 基于surging的木舟平台如何分布式接入设备

    一.概述 上篇文章介绍了木舟通过基于木舟平台浅谈surging 的热点KEY的解决方法,那么此篇文章将介绍基于surging的木舟平台如何分布式接入设备. 木舟 (Kayak) 是什么? 木舟(Kay ...

  4. One API 替代品 Chat Nio 安装与使用教程

    有这样一位初中生,他在初一下学期发起了一个项目,专门用来给他的朋友们免费体验 GPT 模型. 到了八年级的暑假,他决定把这个项目开源出来,并且正式命名为 Chat Nio,同时项目的定位为一站式 LL ...

  5. OS之《线程管理》

    进程是系统资源分配的最小单位,线程是最小的执行单位. 然而,现在的高级设计底层还是基于这个理论基础实现的.比如java的线程,还有最新版本的JDK的协程都是在为了更好的让CPU执行任务. 线程是为了使 ...

  6. R数据分析:PLS结构方程模型介绍,论文报告方法和实际操作

    前面给大家写的关于结构方程模型的文章都是基于变量的方差协方差矩阵来探讨变量间关系的,叫做covariance-based SEM,今天给大家介绍一下另外一个类型的SEM,叫做偏最小二乘结构方差模型.一 ...

  7. 第一个 milestone:内推 50 人拿到微软 offer!

    就在昨天,我的一位候选人和我说,他已经通过面试,正在 offer 沟通阶段.由此,我达成第一个小里程碑:成功内推 50 人拿到了微软 offer! 内推了多少人? 从 2019 年年初开始内推,到现在 ...

  8. Powershell 源码批判

    代码里充斥着过程式编程的搞法:比如这里 Utils.PathIsUnc,分散的到处都是 internal static IEnumerable<string> GetDefaultAvai ...

  9. Python中所有子图标签Legend显示详解

    在数据可视化中,图例(legend)是一个非常重要的元素,它能够帮助读者理解图表中不同元素的含义.特别是在使用Python进行可视化时,matplotlib库是一个非常强大的工具,能够轻松创建包含多个 ...

  10. 【Mybatis】学习笔记01:连接数据库,实现增删改

    需要数据库SQL的请跳转到文末 哔哩哔哩 萌狼蓝天 [转载资料][尚硅谷][MyBatis]2022版Mybatis配套MD文档 [Mybatis]学习笔记01:连接数据库,实现增删改 [Mybati ...