题目链接  2017 CCPC Hangzhou  Problem E

题意  给定一棵树,每个点有一个权值,现在我们可以选一些连通的点,并且把这点选出来的点的权值相加,得到一个和。

求$[1, m]$里面哪些值可以被表示成选出来的点的权值和。用$01$序列的方式输出。

重现赛赛场上的我英勇无畏,大胆做$3000$次FFT合并两个bitset表示的答案。

然后TLE到结束(活该)

其实这个题确实要用bitset,关键是能不能把合并两个bitset转化成合并一个数和一个bitset。

考虑点分治。我们先选出树的重心,然后考虑一定要选这个点的答案。

然后这道题的关键来了。

假设我选择了某个点,那么我必须选择这个点的父亲。

现在开始递归这棵树。每次递归到一个点,这个点的bitset初值化为父亲结点表示的bitset右移$w[x]$位。

他的意义是,当前这个点如果选了,那么他的父亲必选,那也就是求他父亲当前求得的答案集合结合这个点的权值的答案。

然后回溯的时候,父亲表示的bitset要或上儿子表示的bitset。

这是因为在后面这个父亲的其他儿子求解的之后,要用到这个父亲已经求得的信息,并结合起来,

通过这个父亲起到连接的效果。

最后以当前这个分治中心为根的树的答案就是分治中心表示的bitset。

那么接下来继续往下递归求解就可以了。

时间复杂度$O(nlogn \cdot \frac{m}{w})$

#include <bits/stdc++.h>

using namespace std;

#define rep(i, a, b)	for (int i(a); i <= (b); ++i)
#define dec(i, a, b) for (int i(a); i >= (b); --i) const int N = 3e3 + 10;
const int M = 1e5 + 10; int T;
int n, m;
int et;
int f[N], w[N], sz[N], vis[N];
int root, sum;
int cnt;
vector <int> v[N];
bitset <M> b[N], ans; void getroot(int x, int fa){
sz[x] = 1;
f[x] = 0; for (auto u : v[x]){
if (u == fa || vis[u]) continue;
getroot(u, x);
sz[x] += sz[u];
f[x] = max(f[x], sz[u]);
} f[x] = max(f[x], sum - sz[x]);
if (f[x] < f[root]) root = x;
} void calc(int x, int fa, int now){
cnt ^= 1;
b[cnt] = b[cnt ^ 1];
b[cnt] |= (b[cnt ^ 1] << now);
for (auto u : v[x]){
if (vis[u] || u == fa) continue;
calc(u, x, w[u]);
}
} void dp(int x, int fa){
b[x] = b[fa] << w[x];
for (auto u : v[x]){
if (vis[u] || u == fa) continue;
dp(u, x);
b[x] |= b[u];
}
} void calcsize(int x, int fa){
sz[x] = 1;
b[x].reset();
for (auto u : v[x]){
if (vis[u] || u == fa) continue;
calcsize(u, x);
sz[x] += sz[u];
}
} void solve(int x){
++et;
vis[x] = 1; b[0].reset();
b[0].set(0);
calcsize(x, 0);
dp(x, 0);
ans |= b[x]; for (auto u : v[x]){
if (vis[u]) continue;
sum = sz[u];
f[0] = n + 1;
getroot(u, root = 0);
solve(root);
}
} int main(){ scanf("%d", &T);
while (T--){
ans.reset();
scanf("%d%d", &n, &m);
rep(i, 0, n + 1) v[i].clear(); et = 0;
rep(i, 2, n){
int x, y;
scanf("%d%d", &x, &y);
v[x].push_back(y);
v[y].push_back(x);
} rep(i, 1, n) scanf("%d", w + i); memset(vis, 0, sizeof vis);
f[0] = n + 1;
sum = n;
getroot(1, root = 0);
solve(root); rep(i, 1, m) if (ans[i]) putchar(49);
else putchar(48);
putchar(10);
} return 0;
}

HDU 6268 Master of Subgraph (2017 CCPC 杭州 E题,树分治 + 树上背包)的更多相关文章

  1. HDU 6271 Master of Connected Component(2017 CCPC 杭州 H题,树分块 + 并查集的撤销)

    题目链接  2017 CCPC Hangzhou Problem H 思路:对树进行分块.把第一棵树分成$\sqrt{n}$块,第二棵树也分成$\sqrt{n}$块.    分块的时候满足每个块是一个 ...

  2. hdu 6268 Master of Subgraph(点分治+bitset)

    You are given a tree with n nodes. The weight of the i-th node is wi. Given a positive integer m, no ...

  3. 2017 ccpc哈尔滨 A题 Palindrome

    2017 ccpc哈尔滨 A题 Palindrome 题意: 给一个串\(T\),计算存在多少子串S满足\(S[i]=S[2n−i]=S[2n+i−2](1≤i≤n)\) 思路: 很明显这里的回文串长 ...

  4. HDU 6270 Marriage (2017 CCPC 杭州赛区 G题,生成函数 + 容斥 + 分治NTT)

    题目链接  2017 CCPC Hangzhou Problem G 题意描述很清晰. 考虑每个家庭有且仅有$k$对近亲的方案数: $C(a, k) * C(b, k) * k!$ 那么如果在第$1$ ...

  5. 2017 CCPC杭州 题解

    2017CCPC杭州题目PDF Problem A. Super-palindrome 题解: 给你一个字符串,每一步可以将一个字符替换为另一个字符,问你最少多少步可以使得,该字符串任意奇数子串为回文 ...

  6. 算法学习分析-点分治 HDU 6269 Master of Subgraph

    首先给出定义 点分治是一种处理树上路径的工具 挂出一道题目来:Master of Subgraph 这道题目让你求所有联通子图加和所能产生数字,问你1到m之间,那些数字可以被产生 这道题目,假如我们利 ...

  7. 2017 CCPC 杭州 流水账

    day0: 队内训练ccpc 秦皇岛,敝校自己出的题,感觉一个星期没怎么写代码,手生得很,不出意料被打飞了. day1 (热身赛): 热身赛还算顺利,A题看有的队几分钟就草过去了,还以为又是西安ICP ...

  8. 2017 CCPC 杭州 HDU6273J 区间修改(线段树&差分数组)

    http://acm.hdu.edu.cn/downloads/CCPC2018-Hangzhou-ProblemSet.pdf 解析 线段树区间延迟更新 或 差分数组 两个数   统计2和3的最少的 ...

  9. 2017 CCPC 杭州 HDU6265B 积性函数

    题目链接 http://acm.hdu.edu.cn/downloads/CCPC2018-Hangzhou-ProblemSet.pdf B题 数论题      h(n)=∑ d|n φ(d) × ...

随机推荐

  1. USACO Section1.3 Barn Repair 解题报告

    barn1解题报告 —— icedream61 博客园(转载请注明出处)---------------------------------------------------------------- ...

  2. 常用模块(time)

    import time # data = time.time() # 获取时间戳# data = time.localtime() # 获取操作系统时间,也称本地时间,可传入时间戳# data = t ...

  3. Nodejs的那些事

    前言: Node.js实际上是算是个前端开发,但是我们要做APP自动化涉及到 node.js ,也需要它来帮我们安装一些 packing 一.Node.js安装 1.安装Node.js:立即下载 2. ...

  4. isinstance()判断数据类型

    判断数据类型isinstance()l=[1,2,3]print(isinstance(l,list))#括号里面第一个填需要判断的数据,第二个是判断条件

  5. eclipse把jar包引入项目的两种方法

    方法一: build path引入jar包 方法二: 把jar包放入lib文件夹 区别: 把jar包放入lib文件夹,以后把程序发给别人,别人可以直接运行而无需再自己添加jar包 总结: 1.有时即使 ...

  6. [C++] 拓展属性

    inline函数 函数重载 占位参数和默认参数 /*__________________________________________________________________ 背景: C++ ...

  7. 搭建 MongoDB分片(sharding) / 分区 / 集群环境

    1. 安装 MongoDB 三台机器 关闭防火墙 systemctl stop firewalld.service 192.168.252.121 192.168.252.122 192.168.25 ...

  8. Could not automatically select an Xcode project. Specify one in your Podfile like so

    需要将Podfile文件放置在根目录下,而不能放置在项目的里面. 更改路径即可

  9. 发现JDK的3个bug

    1.Annotation引用非空enum数组返回空数组 首次发现时的环境:JDK 1.8 首次发现所在项目:APIJSON 测试用例: public enum RequestRole { /**未登录 ...

  10. TJOI2018游记

    D1T1 - 数学计算 直接用线段树/平衡树维护所有数的积即可.我思想僵化写了一个数学方法...应该是能做\(\bmod\)所有数的乘除法. 时间复杂度\(O(nlogn)\). D1T2 - 智力竞 ...