[Apio2012]dispatching 主席树做法
bzoj 2809: [Apio2012]dispatching
http://www.lydsy.com/JudgeOnline/problem.php?id=2809
Description
Input
Output
Sample Input
0 3 3
1 3 5
2 2 2
1 2 4
2 3 1
Sample Output
HINT
如果我们选择编号为 1的忍者作为管理者并且派遣第三个和第四个忍者,薪水总和为 4,没有超过总预算。
因为派遣了2 个忍者并且管理者的领导力为 3,用户的满意度为 6 ,是可以得到的用户满意度的最大值。
题目剖析:
ans=领导力*个数
领导力:枚举每个忍者
个数:当然是需要薪水少的先派遣啦
所以,节点的领导力为a,在以这个节点为根的子树中(包括这个节点),从小到大选尽可能多的节点s,满足节点的薪水和<=总预算,最大化a*s
这道题还可以用左偏树来做,但蒟蒻一枚只会主席树 ,所以这里只写主席树做法
推荐黄学长的博客,维护子树大根堆,用可并堆将时间复杂度降为nlongn http://hzwer.com/5682.html
数据结构:主席树
辅助:dfs序(开始没想到)
以忍者的dfs序为下标,薪水为区间建立主席树
首先说说dfs序
1、为什么需要dfs序?
因为派遣忍者的范围是以这个忍者为根的子树。
假设当前节点为i,遍历到时令 in[i]=a, 表示i是第a个被遍历。以i为节点的子树遍历完后,令out[i]=b,b为当前所有已遍历节点总数
那么这个节点可以派遣的范围是dfs序[a,b](包括a,b)之间的忍者
所dfs序的作用是锁定查找区间
2、如何构造dfs序?(自己也没想到)
令t表示当前已遍历节点总数,每遍历一个节点t++,设当前节点为i,遍历到i是in[i]=t,i退出时out[i]=t
在构造dfs序时,顺便记录下dfs序为i的忍者的薪水、领导力,然后按照dfs序在主席树中插入节点
最后是查找
枚举每个忍者,查找他的派遣范围最多能派遣多少个忍者
如果做过[CQOI2015]任务查询系统,会知道那道题在查询查到叶子节点时,对答案的贡献是sum[l]/count[l]*k
本题查询的是数量,到叶子节点时有没有坑呢?
有。叶子节点对答案有贡献的忍者数量应该是 min(薪水预算/派遣这个节点的忍者所需薪水,这个节点的忍者个数)
这里的节点是主席树中以薪水为区间建立的节点
还有就是要注意答案用long long
#include<cstdio>
#include<algorithm>
#define N 100001
using namespace std;
int n,m,money[N],lead[N],front[N],next[N];
int hash[N],cnt_money;
int in[N],out[N],t;
int root[N],tot,lc[N*],rc[N*],cnt[N*];
long long sum[N*],ans,dispatch;
struct node
{
int mon,lea;
}e[N];
inline void add(int u,int v)
{
next[v]=front[u];
front[u]=v;
}
inline int dfs(int r)
{
in[r]=++t;
e[t].lea=lead[r];e[t].mon=hash[r];
if(front[r])
for(int i=front[r];i;i=next[i])
dfs(i);
out[r]=t;
}
void discrete()
{
sort(money+,money+n+);
cnt_money=unique(money+,money+n+)-(money+);
for(int i=;i<=n;i++) hash[i]=lower_bound(money+,money+cnt_money+,hash[i])-money;
}
inline void insert(int pre,int & now,int l,int r,int w)
{
sum[now=++tot]=sum[pre]+money[w];
cnt[now]=cnt[pre]+;
if(l==r) return;
int mid=l+r>>;
if(w<=mid)
{
rc[now]=rc[pre];
insert(lc[pre],lc[now],l,mid,w);
}
else
{
lc[now]=lc[pre];
insert(rc[pre],rc[now],mid+,r,w);
}
}
inline int query(int x,int y,int l,int r,long long k)
{
if(l==r) return min(k/(long long)money[l],(long long)(cnt[y]-cnt[x]));
int mid=l+r>>;
long long tmp=sum[lc[y]]-sum[lc[x]];
if(k<=tmp) return query(lc[x],lc[y],l,mid,k);
else return cnt[lc[y]]-cnt[lc[x]]+query(rc[x],rc[y],mid+,r,k-tmp);
}
int main()
{
scanf("%d%lld",&n,&m);
int b,c,l;
for(int i=;i<=n;i++)
{
scanf("%d%d%d",&b,&money[i],&lead[i]);
hash[i]=money[i];
add(b,i);
}
discrete();
dfs(front[]);
for(int i=;i<=n;i++) insert(root[i-],root[i],,cnt_money,e[i].mon);
for(int i=;i<=n;i++)
{
int l=in[i],r=out[i];//l:节点i的dfs序,r节点i可以派遣的最靠后的忍者的dfs序
dispatch=query(root[l-],root[r],,cnt_money,m);
ans=max(ans,(long long)lead[i]*dispatch);
}
printf("%lld",ans);
}
这道题做的时候
1、没有想到用dfs序锁定查找区间
2、ans=max(忍者的领导力*派遣忍者个数),没有想到可以枚举每个忍者,这样就解决了领导力的问题。对答案的分解能力欠缺
3、答案没用long long,开始做的时候还想着答案用long long,想先写出来过了样例后再改,结果忘了
4、查询时,如果当前区间为[x,y],要到节点的右孩子去查询,返回的答案应是区间左右端点左孩子的cnt差,而不是y的cnt。这里与任务查询系统混了,因为任务查询系统查询的区间是[0,y],0可以省略,所以可以直接用y的cnt、sum
[Apio2012]dispatching 主席树做法的更多相关文章
- BZOJ 2809: [Apio2012]dispatching [主席树 DFS序]
传送门 题意:查询树上根节点值*子树中权值和$\le m$的最大数量 最大值是多少 求$DFS$序,然后变成区间中和$\le m$最多有几个元素,建主席树,然后权值线段树上二分就行了 $WA$:又把边 ...
- Luogu P1552 [APIO2012]派遣 主席树
题目链接 Click Here 这个题好像大多数人用的都是左偏树啊?这里我来贡献一发主席树的解法. 把题目中的问题抽象出来,其实就是询问每一个点的子树中,工资前\(tot_i\)大的点,使它们的和满足 ...
- BZOJ - 2809 dispatching 主席树+dfs序
在一个忍者的帮派里,一些忍者们被选中派遣给顾客,然后依据自己的工作获取报偿.在这个帮派里,有一名忍者被称之为 Master.除了 Master以外,每名忍者都有且仅有一个上级.为保密,同时增强忍者们的 ...
- SPOJ DQUERY D-query(主席树)
题目 Source http://www.spoj.com/problems/DQUERY/en/ Description Given a sequence of n numbers a1, a2, ...
- BZOJ3473:字符串(后缀数组,主席树,二分,ST表)
Description 给定n个字符串,询问每个字符串有多少子串(不包括空串)是所有n个字符串中至少k个字符串的子串? Input 第一行两个整数n,k. 接下来n行每行一个字符串. Output 一 ...
- 初识主席树_Prefix XOR
主席树刚接触觉得超强,根本看不懂,看了几位dalao的代码后终于理解了主席树. 先看一道例题:传送门 题目大意: 假设我们预处理出了每个数满足条件的最右边界. 先考虑暴力做法,直接对x~y区间暴枚,求 ...
- BZOJ 2161 布娃娃 (主席树)
题面 想了一个主席树做法 我们把每个区间的两个端点拆开 对$L,R$分别从小到大排序,分别从左到右依次把对应标号的$c_{i}$插入到权值主席树里 每次查询$p_{i}$,在排序后的$L,R$数组上分 ...
- LOJ 6435 「PKUSC2018」星际穿越——DP+倍增 / 思路+主席树
题目:https://loj.ac/problem/6435 题解:https://www.cnblogs.com/HocRiser/p/9166459.html 自己要怎样才能想到怎么做呢…… dp ...
- zoj2112 Dynamic Rankings (主席树 || 树套树)
The Company Dynamic Rankings has developed a new kind of computer that is no longer satisfied with t ...
随机推荐
- 第一次spring冲刺第5天
今天进行讨论基础功能的核心代码方面,还有简单的讨论继续关于界面的美化, 计算生成的答案功能 public class Core {// char[]h={'+','-','*','/'};int re ...
- Fast Packet Processing - A Survey
笔记是边读边写的旁注,比较乱,没有整理就丢上来了. 可以说不仅要说fast packet process servey,也同时是一篇packet process的综述了.packet processi ...
- Software Defined Networking(Week 2, part 3)
Control of Packet-switch Network 我们已经讨论过中心控制网络的原理,但主要是以电话网络做模型的.现在我们来看看对于分组交换网络的控制是如何改进的. Why Separa ...
- self和super关键字介绍
1.self和super OC提供两个保留字self 和 super ,用在方法定义中 OC语言中的self, 就相当于C++和Java中的this指针,学会使用self 首先要搞清楚属性这一概念以及 ...
- [转帖] 红帽8.0 beta版本发布 内核新版本 4.18
Red Hat Enterprise Linux 8 Beta 现已发布! https://www.oschina.net/news/101870/red-hat-enterprise-linux-8 ...
- poj 1185(状态压缩DP)
poj 1185(状态压缩DP) 题意:在一个N*M的矩阵中,‘H'表示不能放大炮,’P'表示可以放大炮,大炮能攻击到沿横向左右各两格,沿纵向上下各两格,现在要放尽可能多的大炮使得,大炮之间不能相互 ...
- 我的虚拟机中的 centOS 连不了网了
网上的办法试过了,查看虚拟机的网络配置,是 NET的, 也 cd 到/etc/sysconfig/network-script/ifcfg-eth0 里面看了,onboot 本来就是 yes,要不然我 ...
- MySQL 双主问题集
最近试用MySQL高可用方案,需要配MySQL双主,对期间遇到的问题做下记录. 1.导出锁表问题 mysqldump 命令增加参数 --skip-opt -q 可避免导出时锁表: 2.导出\导入所有数 ...
- win10与Ubantu双系统:Linux下开启FTP服务器与创建无线热点(实现文件共享)
如何在win系统下使用filelizza这个软件搭建FTP服务器,然后建立一个无线局域网,让平板终端连接以后,访问电脑硬盘的文件. 如果是只在win7环境下,一切都很简单,按照上文提供的教程就可以实现 ...
- Musical Theme POJ - 1743(后缀数组+二分)
求不可重叠最长重复子串 对于height[i]定义为sa[i]和 sa[i-1]的最长公共前缀 这个最长公共前缀的值肯定是最大的 证明: 设rank[j] < rank[k], 则不难证明后缀j ...