Description

Crystal家有一棵树。树上有\(n\)个节点,编号由\(1\)到\(n\)(\(1\)号点是这棵树的根),两点之间距离为1当且仅当它们直接相连。每个点都有各自的权值,第\(i\)号节点的权值为\(value_i\)。Crystal现在指着编号为\(x\)的点问,在以点\(x\)为根的子树中,与点\(x\)距离大于等于\(k\)的所有点的点权和是多少。

Input Format

第\(1\)行两个整数\(n,Q\),分别表示树上点的个数和Crystal有\(Q\)个问题。

第\(2\)行,\(n\)个整数,分别表示\(1\)至\(n\)号点的点权。

接下来的\(n - 1\)行,每行两个整数\(u,v\),表示编号为\(u\)的点与编号为\(v\)的点直接相连。

接下来\(Q\)行,每行两个整数\(x,k\),表示询问在以点\(x\)为根的子树中,与点\(x\)距离大于等于为\(k\)的所有点的点权和是多少。

Output Format

\(Q\)行,每行一个整数,表示对第\(i\)个询问的回答。

Sample Input

5 3

1 1 1 1 1

1 2

1 3

3 4

4 5

1 3

1 2

1 1

Sample Output

1

2

4

Hints

对于\(30\%\)的数据,保证\(n \le 1000, k < 1, Q \le 1000\)。

对于\(60\%\)的数据,保证\(n \le 1000, k < 1000, Q \le 1000\)

对于\(80\%\)的数据,保证\(n \le 1000, k < 1000, Q \le 1000000\);

对于最后\(20\%\)的数据,保证\(n \le 50000, k < 100, Q \le 1000000\);

对于\(100\%\)的数据,保证所有输入数据均为非负整数,且在\(int\)范围内。

这题\(O(NK)\)的做法不难想(用总的减去小于\(K\)的),现在假设\(N,K\)同级怎么做。

首先考虑离线做法,我们可以考虑按照询问最深的深度从小到大一层层加点,答案还是用总的减去小于\(K\)的。

再考虑在线所做法,我们可以先处理出dfs序和子树和,然后对于树的每层开一个vector,vector中记录该层点的编号,按dfs序排序。对于每个询问\(x,k\),我们只需要跳到\(dep[x]+k\)层的vector中,找到在\(x\)子树中的点,且一定是段连续区间,二分即可。现在只需要对该区间求子树和的和即可。

代码是\(O(NK)\)的

    #include<cstdio>
#include<cstdlib>
#include<iostream>
#include<cstring>
#include<vector>
using namespace std; typedef long long ll;
#define maxn (50010)
int cnt = 1,side[maxn],toit[maxn*2],next[maxn*2],val[maxn],N,Q,mxk;
int tx[maxn*20],tk[maxn*20],num[20],len; ll sum[maxn]; vector <ll> res[maxn]; inline int read()
{
char ch; int f = 1,ret = 0;
do ch = getchar(); while (!(ch >= '0'&&ch <= '9')&&ch != '-');
if (ch == '-') f = -1,ch = getchar();
do ret = ret*10+ch-'0',ch = getchar(); while (ch >= '0'&&ch <= '9');
return ret*f;
} inline void add(int a,int b) { next[++cnt] = side[a]; side[a] = cnt; toit[cnt] = b; }
inline void ins(int a,int b) { add(a,b); add(b,a); } inline void dfs(int now,int fa)
{
for (int i = 0;i <= mxk;++i) res[now].push_back(val[now]);
sum[now] = val[now];
for (int i = side[now];i;i = next[i])
{
if (toit[i] == fa) continue;
dfs(toit[i],now);
sum[now] += sum[toit[i]];
for (int j = 0;j < mxk;++j)
res[now][j+1] += res[toit[i]][j];
}
} inline void print(ll a)
{
do num[++len] = a%10,a /= 10; while (a);
while (len) putchar('0'+num[len--]);
puts("");
} int main()
{
//freopen("a.in","r",stdin);
//freopen("a.out","w",stdout);
N = read(); Q = read();
for (int i = 1;i <= N;++i) val[i] = read();
for (int i = 1;i < N;++i) ins(read(),read());
for (int i = 1;i <= Q;++i) tx[i] = read(),tk[i] = read(),mxk = max(mxk,tk[i]);
dfs(1,0);
// print(123456LL);
// print(0LL);
// print(12LL);
for (int i = 1;i <= Q;++i)
{
if (!tk[i]) //cout << sum[tx[i]] << endl;
print(sum[tx[i]]);
else //cout << sum[tx[i]]-res[tx[i]][tk[i]-1] << endl;
print(sum[tx[i]]-res[tx[i]][tk[i]-1]);
}
//fclose(stdin); fclose(stdout);
return 0;
}

sjtu1591 Count On Tree的更多相关文章

  1. leetcode面试准备:Count Complete Tree Nodes

    1 题目 Given a complete binary tree, count the number of nodes. In a complete binary tree every level, ...

  2. leetcode 958. Check Completeness of a Binary Tree 判断是否是完全二叉树 、222. Count Complete Tree Nodes

    完全二叉树的定义:若设二叉树的深度为h,除第 h 层外,其它各层 (1-h-1) 的结点数都达到最大个数,第 h 层所有的结点都连续集中在最左边,这就是完全二叉树. 解题思路:将树按照层进行遍历,如果 ...

  3. 完全二叉树的节点个数 Count Complete Tree Nodes

    2018-09-25 16:36:25 问题描述: 问题求解: 单纯遍历了一遍,emmm,果然TLE. 解题思路就是比较左边树高度和右边树高度,如果相等,那么就是一个满二叉树,返回1 << ...

  4. 【LeetCode】222. Count Complete Tree Nodes 解题报告(Python)

    [LeetCode]222. Count Complete Tree Nodes 解题报告(Python) 标签(空格分隔): LeetCode 作者: 负雪明烛 id: fuxuemingzhu 个 ...

  5. 【刷题-LeetCode】222. Count Complete Tree Nodes

    Count Complete Tree Nodes Given a complete binary tree, count the number of nodes. Note: Definition ...

  6. [LeetCode] Count Complete Tree Nodes 求完全二叉树的节点个数

    Given a complete binary tree, count the number of nodes. Definition of a complete binary tree from W ...

  7. Count Complete Tree Nodes

    Given a complete binary tree, count the number of nodes. Definition of a complete binary tree from W ...

  8. Java for LeetCode 222 Count Complete Tree Nodes

    Given a complete binary tree, count the number of nodes. Definition of a complete binary tree from W ...

  9. leetcode_222 Count Complete Tree Nodes

    题目: Given a complete binary tree, count the number of nodes. Definition of a complete binary tree fr ...

随机推荐

  1. CentOS 7下的软件安装方法及策略

    一些废话 2010年开始正式接触Linux,入门发行版是Ubuntu 10.10,后来过渡到Ubunu 11.04,这其中也尝试了很多其他主流的发行版.进入实验室之后,开始用CentOS 5,然后是C ...

  2. selendroid inspector xpth元素定位记录

    android自动化测试元素定位,目前发现appium官方的uiautomatorviewer一般的元素定位还行,但好多都找不到. 这个时候,可以考虑selendroid的inspector 官网:h ...

  3. MySQL高可用解决方案(MySQL HA Solution)

    http://blog.sina.com.cn/s/blog_7e89c3f501012vtr.html 什么是高可用性?很多公司的服务都是24小时*365天不间断的.比如Call Center.这就 ...

  4. Hadoop源代码分析

    http://wenku.baidu.com/link?url=R-QoZXhc918qoO0BX6eXI9_uPU75whF62vFFUBIR-7c5XAYUVxDRX5Rs6QZR9hrBnUdM ...

  5. vs2010打开vs2012的sln文件

    1.找到**.sln文件,然后选择用记事本打开. 2.最前面找到“Microsoft Visual Studio Solution File, Format Version 12.00  # Visu ...

  6. 解决从linux本地文件系统上传文件到HDFS时的权限问题

    当使用 hadoop fs -put localfile /user/xxx 时提示: put: Permission denied: user=root, access=WRITE, inode=& ...

  7. 2 JavaScript应用开发实践指南

    JavaScript 语言在浏览器中的运用 HTTP请求,加载HTML后根据内容加载CSS等,大部分浏览器默认2个下载链接. HTML元素要尽可能简洁,不需要将Table元素变成多个div, css代 ...

  8. JavaScript DOM编程艺术 - 读书笔记1-3章

    1.JavaScript语法 准备工作 一个普通的文本编辑器,一个Web浏览器. JavaScript代码必须通过Html文档才能执行,第一种方式是将JavaScript代码放到文档<head& ...

  9. c++学习笔记1(c++简介)

    c++和c的不同: 1,c++是c的扩充. 2,在解决问题时思维方式的不同.(c++采用面向对象思维,c面向结构思维) 面向结构思维:将一个大程序拆分成一个个很小的结构.每个结构完成一个或多个功能,所 ...

  10. 转:基于IOS上MDM技术相关资料整理及汇总

    一.MDM相关知识: MDM (Mobile Device Management ),即移动设备管理.在21世纪的今天,数据是企业宝贵的资产,安全问题更是重中之重,在移动互联网时代,员工个人的设备接入 ...