首先给出定义

点分治是一种处理树上路径的工具

挂出一道题目来:Master of Subgraph

这道题目让你求所有联通子图加和所能产生数字,问你1到m之间,那些数字可以被产生

这道题目,假如我们利用暴力的方法去求解的话

实际上是对每个节点进行一次dfs,这样的话会发现复杂度为O(N^2)也就是再9e6左右,再加上常数M/64,复杂度根本不够(9e9)

我们可以利用点分治去优化复杂度

点分治的原理就是树上的路径产生的答案,不是在经过这个节点的就是在不经过这个节点的,那我们找到树的重心的话,就能够算出来经过该点的所有答案,然后依次递归大约logN层,这样复杂度就变成了NlogN的程度,也就是3e4*1e3=3e7加上常数,再加上时间的宽限,完全够用

所以点分治就是树上分治的一种,减小重复计算的东西,不断逐步缩小子树的计算

其中一般来说要开 一个父节点数组,一个儿子数数组,来计算重心的位置

然后就是点分治,跟递归差不多就是要去计算答案

然后就是利用solve函数,每次点分治计算树根,然后依次处理子树的重心节点然后继续递归继续分治

 #include <iostream>
 #include <cstring>
 #include <vector>
 #include <bitset>

 ;
 ;

 int E[MAXN][MAXN];
 int all[MAXN];

 int f[MAXN],son[MAXN],root,tot;
 /*
     分别代表f[x]的两侧孩子数目的最大值,重心的孩子数目最小
     son代表以x为根的孩子数目
     root是被移动的根
     tot是当前根的孩子总数
  */
 bool vis[MAXN];

 std::bitset<MAXM>b[MAXN],ans;

 int n,m,val[MAXN];

 void dfs(int x,int fa){
     /* dfs搜索树,将根移动到树的重心,降低dp层数 */
     f[x] = ;
     /* 标记孩子数目为0 */
     son[x] = ;
     /* 计算孩子数目 */
     ; i <= all[x]; i++)
     {
         int y=E[x][i];
         if(!vis[y] && y!=fa)
         {
             dfs(y,x);
             /* 递归进入子数 */
             f[x] = std::max(f[x],son[y]);
             /* 计算子树中孩子数目最多的子树孩子数 */
             son[x]+=son[y];
             /* 累加孩子数目 */
         }
     }
     f[x]=std::max(f[x],tot-f[x]);
     /* 计算该根节点最大孩子数其余侧的数目中两边的最大值 */
     if(f[x]<f[root]) root=x;
     /* 移动根节点,寻找重心 */
 }

 void getdp(int x, int fa){
     b[x]<<=val[x];
     /* 累加val[x] */
     son[x]=;

     ;i<=all[x];i++)
     {
         int y=E[x][i];
         if(!vis[y] && y!=fa)
         {
             b[y]=b[x];
             getdp(y,x);
             son[x]+=son[y];
             b[x]|=b[y];
         }
     }
 }

 void solve(int x){
     vis[x] = true;
     b[x] = ;
     getdp(x,);
     /* 以某一点为根进行点分治 */
     ans|=b[x];
     /* 累加答案 */

     ;i<=all[x];i++)
     {
         /* 子树递归进行点分治 */
         int y=E[x][i];
         if(!vis[y])
         {
             tot = son[y];
             root = ;
             dfs(y,x);
             /* 先寻找树根 */
             solve(root);
             /* 递归分治 */
         }
     }
 }

 int main(){
     int T;
     std::cin>>T;
     while(T--)
     {
         scanf("%d%d",&n,&m);
         ;i<=n;i++) all[i]=;
         ans.reset();
         memset(vis,,sizeof(vis));
         ;i<n;i++)
         {
             int x,y;
             scanf("%d%d",&x,&y);
             E[x][++all[x]]=y;
             E[y][++all[y]]=x;
         }
         ;i<=n;i++)
         {
             scanf("%d",&val[i]);
         }

         f[] = n+;
         tot=n;
         dfs(,root);
         solve(root);
         ;i<=m;i++)
         {
             printf("%d",(int)ans[i]);
         }
         puts("");
     }
     ;
 }

算法学习分析-点分治 HDU 6269 Master of Subgraph的更多相关文章

  1. HDU 6268 Master of Subgraph (2017 CCPC 杭州 E题,树分治 + 树上背包)

    题目链接  2017 CCPC Hangzhou  Problem E 题意  给定一棵树,每个点有一个权值,现在我们可以选一些连通的点,并且把这点选出来的点的权值相加,得到一个和. 求$[1, m] ...

  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. <2014 05 09> Lucida:我的算法学习之路

    [转载] 我的算法学习之路 关于 严格来说,本文题目应该是我的数据结构和算法学习之路,但这个写法实在太绕口——况且CS中的算法往往暗指数据结构和算法(例如算法导论指的实际上是数据结构和算法导论),所以 ...

  4. 算法学习之C语言基础

    算法学习,先熟悉一下C语言哈!!! #include <conio.h> #include<stdio.h> int main(){ printf(+); getch(); ; ...

  5. Python之路,Day21 - 常用算法学习

    Python之路,Day21 - 常用算法学习   本节内容 算法定义 时间复杂度 空间复杂度 常用算法实例 1.算法定义 算法(Algorithm)是指解题方案的准确而完整的描述,是一系列解决问题的 ...

  6. 二次剩余Cipolla算法学习笔记

    对于同余式 \[x^2 \equiv n \pmod p\] 若对于给定的\(n, P\),存在\(x\)满足上面的式子,则乘\(n\)在模\(p\)意义下是二次剩余,否则为非二次剩余 我们需要计算的 ...

  7. 算法学习之快速排序的C语言实现

    近几天在学习简单算法,今天看了一个快速排序和堆排序,堆排序还没搞懂,还是先把快速排序搞清楚吧 教程网上一艘一大堆,这里选择一个讲的比较通俗的的一个吧: http://blog.csdn.net/mor ...

  8. 3.2_k-近邻算法案例分析

        k-近邻算法案例分析 本案例使用最著名的”鸢尾“数据集,该数据集曾经被Fisher用在经典论文中,目前作为教科书般的数据样本预存在Scikit-learn的工具包中. 读入Iris数据集细节资 ...

  9. Reinforcement Learning Q-learning 算法学习-2

    在阅读了Q-learning 算法学习-1文章之后. 我分析了这个算法的本质. 算法本质个人分析. 1.算法的初始状态是随机的,所以每个初始状态都是随机的,所以每个初始状态出现的概率都一样的.如果训练 ...

随机推荐

  1. 把自定义的decoder加入ffmpeg源码

    第一步: 在libavcodec目录下新建mkdecoder.c,并加入一下代码: /* *实现一个自己的decoder,编码工作其实就是把pkt的数据拷贝到frame *作者:缪国凯(MK) *82 ...

  2. 在Windows下编译WebRTC

    前言 这篇文章的目的在于为你节省生命中宝贵的10小时(甚至更多),或者浪费你10分钟.作为Google更新频繁的大型跨平台基础库,WebRTC的编译一直被人称为噩梦.如果恰巧你偏要在Windows下编 ...

  3. 省选/NOI刷题Day2

    bzoj2616 放一个车的时候相当于剪掉棋盘的一行,于是就可以转移了,中间状态转移dp套dp,推一下即可 bzoj2878 环套树期望dp 手推一下递推式即可 bzoj3295 树状数组套权值线段树 ...

  4. LOJ2305 「NOI2017」游戏

    「NOI2017」游戏 题目背景 狂野飙车是小 L 最喜欢的游戏.与其他业余玩家不同的是,小 L 在玩游戏之余,还精于研究游戏的设计,因此他有着与众不同的游戏策略. 题目描述 小 L 计划进行$n$场 ...

  5. OI省选算法汇总( 转发黄学长博客 )

    [原文链接] http://hzwer.com/1234.html 注 : 蓝色为已学习算法 , 绿色为不熟练算法 , 灰色为未学习算法 1.1 基本数据结构 1. 数组 2. 链表,双向链表 3. ...

  6. Windows 任务管理器中的几个内存概念

    我们使用的大部分 PC 是基于 Intel 微处理器的 x86 和 x64 架构计算机. 因此, 我们面对的 windows 避免不了和 Intel 架构有些设计上的契合. 比如接下来要说到的内存管理 ...

  7. BZOJ4695:最假女选手

    浅谈区间最值操作和历史最值问题:https://www.cnblogs.com/AKMer/p/10225100.html 题目传送门:https://lydsy.com/JudgeOnline/pr ...

  8. Poj 2304 Combination Lock(模拟顺、逆时钟开组合锁)

    一.题目大意 模拟一个开组合的密码锁过程.就像电影你开保险箱一样,左转几圈右转几圈的就搞定了.这个牌子的锁呢,也有它独特的转法.这个锁呢,有一个转盘,刻度为0~39.在正北方向上有一个刻度指针.它的密 ...

  9. ios判断是否为iphone6或iphone6plus代码

    转自:http://blog.csdn.net/lvxiangan/article/details/45288505 #define IS_IPAD (UI_USER_INTERFACE_IDIOM( ...

  10. Qt 按顺序保存多个文件

    void MainWindow::on_pushButtonSnap_clicked() { ]; sprintf(image_name, "%s%d%s", "C:/i ...