在一个忍者的帮派里,一些忍者们被选中派遣给顾客,然后依据自己的工作获取报偿。在这个帮派里,有一名忍者被称之为 Master。除了 Master以外,每名忍者都有且仅有一个上级。为保密,同时增强忍者们的领导力,所有与他们工作相关的指令总是由上级发送给他的直接下属,而不允许通过其他的方式发送。现在你要招募一批忍者,并把它们派遣给顾客。你需要为每个被派遣的忍者 支付一定的薪水,同时使得支付的薪水总额不超过你的预算。另外,为了发送指令,你需要选择一名忍者作为管理者,要求这个管理者可以向所有被派遣的忍者 发送指令,在发送指令时,任何忍者(不管是否被派遣)都可以作为消息的传递 人。管理者自己可以被派遣,也可以不被派遣。当然,如果管理者没有被排遣,就不需要支付管理者的薪水。你的目标是在预算内使顾客的满意度最大。这里定义顾客的满意度为派遣的忍者总数乘以管理者的领导力水平,其中每个忍者的领导力水平也是一定的。写一个程序,给定每一个忍者 i的上级 Bi,薪水Ci,领导力L i,以及支付给忍者们的薪水总预算 M,输出在预算内满足上述要求时顾客满意度的最大值。
 
$1≤N≤100000$ 忍者的个数;
$1≤M≤1000000000$ 薪水总预算; 
$0≤B_i < i$ 忍者的上级的编号;
$1≤C_i≤M$   忍者的薪水;
$1≤L_i≤1000000000$   忍者的领导力水平。
 
题解:
  问题等价于,每一个点最多能在子树上取多少点,花费小于M?
  显然的是,我们的策略一定是从最小的点开始取,直到超过$M$为止,这显然是贪心的过程
  然后得枚举所有点,总复杂度$O(n^2)$
  然后考虑优化,
  我们把问题"询问一个点,其子树上之和大于$M$的$x$个权值最小的点"
  转化成"询问一个区间,查询一个区间上和小于等于$M$的$x$个最小的元素"
  显然,后者是主席树可以解决的,单次操作的复杂度就从$O(n)$变成了$O(log(n))$
  然后我们用$dfs$序将树拍扁,变成序列即可解决问题,就完成了优化
  具体过程就是
  $dfs(master)$得到$dfs$序,离散化花费
  主席树保存两个值,一个是点数,一个是花费,单点更新,区间求和
  按照$dfs$序列插入权值,构造n颗树
  最后,遍历每个点,每个点等价于一个区间询问,枚举所有点就能得到最大值
  (这道题网上的主席树题解代码真是乱写,反正我是没懂他们是怎么写的,AC是AC了,莫名其妙多此一举的操作一大堆)
  1. #include <bits/stdc++.h>
  2. #define nd seg[now]
  3. #define ndp seg[pre]
  4. #define mid ((s+t)>>1)
  5. #define ll long long
  6. using namespace std;
  7. const int maxn=1e5+10;
  8. const int maxm=1e6+10;
  9. const int INF=0x3f3f3f3f;
  10. int casn,n,k;
  11. ll m;
  12. struct node2{
  13. int to,next;
  14. }e[maxn];
  15. int head[maxn],nume;
  16. int tin[maxn],tout[maxn];
  17. ll cost[maxn],pos[maxn],power[maxn];
  18. int rt[maxn],size,cnt,pre[maxn],dfn[maxn];
  19. int cmp(int a,int b){
  20. return cost[a]<cost[b];
  21. }
  22. inline void add(int a,int b){
  23. e[++nume]=(node2){b,head[a]};
  24. head[a]=nume;
  25. }
  26. void dfs(int now){
  27. tin[now]=++cnt;
  28. dfn[cnt]=now;
  29. for(int i=head[now];i;i=e[i].next){
  30. dfs(e[i].to);
  31. }
  32. tout[now]=cnt;
  33. }
  34. struct node{
  35. int l,r;ll sum,cnt;
  36. }seg[maxn*20];
  37. void maketree(int s=1,int t=n,int &now=rt[0]){
  38. now=++size;nd=(node){s,t,0,0};
  39. if(s==t) return ;
  40. maketree(s,mid,nd.l);maketree(mid+1,t,nd.r);
  41. }
  42. void update(int &now,int pre,int k,ll cost,int s=1,int t=n){
  43. now=++size;nd=ndp,nd.sum+=cost,nd.cnt++;
  44. if(s==t) return ;
  45. if(k<=mid)update(nd.l,ndp.l,k,cost,s,mid);
  46. else update(nd.r,ndp.r,k,cost,mid+1,t);
  47. }
  48. ll query(int ndl,int ndr,ll k,int s=1,int t=n){
  49. if(seg[ndr].sum-seg[ndl].sum<=k) return seg[ndr].cnt-seg[ndl].cnt;
  50. if(s==t) return min(seg[ndr].cnt-seg[ndl].cnt,k/pos[s]);
  51. ll sum=seg[seg[ndr].l].sum-seg[seg[ndl].l].sum;
  52. if(k>=sum) return query(seg[ndl].r,seg[ndr].r,k-sum,mid+1,t)+seg[seg[ndr].l].cnt-seg[seg[ndl].l].cnt;
  53. else return query(seg[ndl].l,seg[ndr].l,k,s,mid);
  54. }
  55. #undef mid
  56. int main(){
  57. scanf("%d%lld",&k,&m);
  58. int master;
  59. for(int i=1;i<=k;i++){
  60. scanf("%d%lld%lld",pre+i,cost+i,power+i);
  61. if(pre[i]==0)master=i;
  62. else add(pre[i],i);
  63. pos[i]=cost[i];
  64. }
  65. sort(pos+1,pos+1+k);
  66. n=unique(pos+1,pos+1+k)-(pos+1);
  67. dfs(master);
  68. maketree();
  69. for(int i=1;i<=k;i++){
  70. int id=lower_bound(pos+1,pos+1+n,cost[dfn[i]])-pos;
  71. update(rt[i],rt[i-1],id,cost[dfn[i]]);
  72. }
  73. ll ans=0;
  74. for(int i=1;i<=k;i++){
  75. ans=max(ans,power[i]*query(rt[tin[i]-1],rt[tout[i]],m));
  76. }
  77. printf("%lld\n",ans);
  78. return 0;
  79. }

  

 
  

BZOJ - 2809 dispatching 主席树+dfs序的更多相关文章

  1. BZOJ 2809: [Apio2012]dispatching [主席树 DFS序]

    传送门 题意:查询树上根节点值*子树中权值和$\le m$的最大数量 最大值是多少 求$DFS$序,然后变成区间中和$\le m$最多有几个元素,建主席树,然后权值线段树上二分就行了 $WA$:又把边 ...

  2. 51 nod 1681 公共祖先 (主席树+dfs序)

    1681 公共祖先 基准时间限制:1 秒 空间限制:131072 KB 分值: 80 难度:5级算法题   有一个庞大的家族,共n人.已知这n个人的祖辈关系正好形成树形结构(即父亲向儿子连边). 在另 ...

  3. 【BZOJ1803】Spoj1487 Query on a tree III 主席树+DFS序

    [BZOJ1803]Spoj1487 Query on a tree III Description You are given a node-labeled rooted tree with n n ...

  4. bzoj 3772 精神污染 主席树+dfs序

    精神污染 Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 637  Solved: 177[Submit][Status][Discuss] Descri ...

  5. 【SPOJ】10628. Count on a tree(lca+主席树+dfs序)

    http://www.spoj.com/problems/COT/ (速度很快,排到了rank6) 这题让我明白了人生T_T 我知道我为什么那么sb了. 调试一早上都在想人生. 唉. 太弱. 太弱. ...

  6. BZOJ3772 精神污染 主席树 dfs序

    欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ3772 题意概括 给出一个树,共n个节点. 有m条互不相同的树上路径. 现在让你随机选择2条路径,问 ...

  7. BZOJ5338[TJOI2018]xor——主席树+dfs序

    题目描述 现在有一颗以1为根节点的由n个节点组成的树,树上每个节点上都有一个权值vi. 现在有Q 次操作,操作如下: 1  x y    查询节点x的子树中与y异或结果的最大值 2 x y z     ...

  8. BZOJ3545&3551[ONTAK2010]Peaks——kruskal重构树+主席树+dfs序+树上倍增

    题目描述 在Bytemountains有N座山峰,每座山峰有他的高度h_i.有些山峰之间有双向道路相连,共M条路径,每条路径有一个困难值,这个值越大表示越难走,现在有Q组询问,每组询问询问从点v开始只 ...

  9. Codeforces 893F(主席树+dfs序)

    在子树内和距离不超过k是一个二维限制,容易想到主席树,但主席树显然没法查最小值,因为不满足区间可减.kdtree和二维线段树可以干这事,但肯定会T飞.但事实上我们的问题有一个特殊性:对某个点x,查询其 ...

随机推荐

  1. float clearfix

    Float float 属性的原本作用是: 为了实现文字环绕效果 float 父元素高度塌陷实现文字环绕效果 float 固定一列宽的自适应布局 float 多列布局` float 固定一列宽的自适应 ...

  2. SpringMVC的概念和图解

    1.概念 Spring MVC起步:慕课网视频 SpringMVC架构浅析:参考 Spring详解(一)------概述 Spring架构简单描述 2.图片

  3. JavaScript中的this指向规则

    首先,JavaScript的this指向问题并非传说中的那么难,不难的是机制并不复杂,而被认为不好理解的是逻辑关系和容易混淆的执行上下文.这篇博客也就会基于这两个不好理解的角度来展开,如要要严格的来对 ...

  4. FeignClient调用POST请求时查询参数被丢失的情况分析与处理

    前言 本文没有详细介绍 FeignClient 的知识点,网上有很多优秀的文章介绍了 FeignCient 的知识点,在这里本人就不重复了,只是专注在这个问题点上. 查询参数丢失场景 业务描述: 业务 ...

  5. Web 学习笔记(一)百度统计

    一.百度统计是什么? 百度统计是百度推出的一款免费的专业网站流量分析工具,能够告诉用户访客是如何找到并浏览用户的网站,在网站上做了些什么,有了这些信息,可以帮助用户改善访客在用户的网站上的使用体验,不 ...

  6. Ant和Maven

    Ant和Maven都是基于Java的构建(build)工具.理论上来说,有些类似于(Unix)C中的make ,但没有make的缺陷.Ant是软件构建工具,Maven的定位是软件项目管理和理解工具. ...

  7. Structured Streaming + Kafka Integration Guide 结构化流+Kafka集成指南 (Kafka broker version 0.10.0 or higher)

    用于Kafka 0.10的结构化流集成从Kafka读取数据并将数据写入到Kafka. 1. Linking 对于使用SBT/Maven项目定义的Scala/Java应用程序,用以下工件artifact ...

  8. vue input添加回车触发

    普通vue input @keyup.enter="onSubmit" element el-input @keyup.enter.native="onSubmit&qu ...

  9. C#窗口闪烁问题解决

    https://www.cnblogs.com/AndyDai/p/5203798.html 开发WinForm 程序时经常会遇到闪屏的问题,这会给用户造成很差的使用体验,所以必须妥善解决好这个问题. ...

  10. 【四】Java虚拟机内存区域初识

     一.线程独占区  1.程序计数器 程序计数器是一块处于线程独占区较小的内存空间,它可以看是当前线程所执行的字节码的行号指示器. 如果线程执行的是Java方法,这个计数器记录的是正在执行的虚拟机字节码 ...