Description

在一个忍者的帮派里,一些忍者们被选中派遣给顾客,然后依据自己的工作获取报偿。

在这个帮派里,有一名忍者被称之为Master。除了Master以外,每名忍者都有且仅有一个上级。为保密,同时增强忍者们的领导力,所有与他们工作相关的指令总是由上级发送给他的直接下属,而不允许通过其他的方式发送。

现在你要招募一批忍者,并把它们派遣给顾客。你需要为每个被派遣的忍者支付一定的薪水,同时使得支付的薪水总额不超过你的预算。另外,为了发送指令,你需要选择一名忍者作为管理者,要求这个管理者可以向所有被派遣的忍者发送指令,在发送指令时,任何忍者(不管是否被派遣)都可以作为消息的传递人。管理者自己可以被派遣,也可以不被派遣。当然,如果管理者没有被排遣,你就不需要支付管理者的薪水。

你的目标是在预算内使顾客的满意度最大。这里定义顾客的满意度为派遣的忍者总数乘以管理者的领导力水平,其中每个忍者的领导力水平也是一定的。

写一个程序,给定每一个忍者i的上级Bi,薪水Ci,领导力Li,以及支付给忍者们的薪水总预算M,输出在预算内满足上述要求时顾客满意度的最大值。

Input

第一行包含两个整数N和M,其中N表示忍者的个数,M表示薪水的总预算。

接下来N行描述忍者们的上级、薪水以及领导力。其中的第i行包含三个整数Bi,Ci,Li分别表示第i个忍者的上级,薪水以及领导力。Master满足Bi=0,并且每一个忍者的老板的编号一定小于自己的编号Bi<i。

Output

输出一个数,表示在预算内顾客的满意度的最大值。

Sample Input

5 4
0 3 3
1 3 5
2 2 2
1 2 4
2 3 1

Sample Output

6

Hint

1 ≤ N ≤ 100,000 忍者的个数;

1 ≤ M ≤ 1,000,000,000 薪水总预算;

0 ≤ Bi < i 忍者的上级的编号;

1 ≤ Ci ≤ M 忍者的薪水;

1 ≤ Li ≤ 1,000,000,000 忍者的领导力水平。

对于 30%的数据,N ≤ 3000。

题解

有用平衡树做的,这里给出可并堆的写法。

首先我们由贪心的思想我们可以知道:对于每个点,我们把每个以他为代表的子树里的所有点,从小到大排好序。然后一直选小的,直到不满足条件为止。

我们从叶子到根进行合并即可。用按$DFS$逆序操作。

我们将以其为根的子树放在一个按照花销为关键词的大根堆里面,一直$pop$掉堆顶,直到整个堆里面的花费$<=n$

再将这个堆中剩余的元素并到其父亲所在的堆中,继续操作。

最后统计一下答案即可。

 #include<set>
#include<map>
#include<cmath>
#include<ctime>
#include<queue>
#include<stack>
#include<vector>
#include<cstdio>
#include<string>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#define LL long long
#define RE register
#define IL inline
using namespace std;
const LL N=;
const LL INF=~0u>>; IL LL Max(const LL &a,const LL &b) {return a>b ? a:b;} struct node
{
LL key,cost,dist;
node *l,*r;
LL ldist() {return l ? l->dist:;}
LL rdist() {return r ? r->dist:;}
}H[N+],*root[N+];
LL father[N+],sum[N+],remain[N+];
node* Merge(node* a,node* b); LL n,m;
LL b,c,l; struct tt
{
LL to,next;
}edge[N+];
LL path[N+],top;
IL void Add(LL u,LL v);
LL Rank[N+],tail;
void Dfs(LL r,LL father); int main()
{
scanf("%lld%lld",&n,&m);
for (RE LL i=;i<=n;i++)
{
scanf("%lld%lld%lld",&b,&c,&l);
father[i]=b;
remain[i]=;
sum[i]=H[i].cost=c;
H[i].key=l;
root[i]=H+i;
Add(b,i);
}
Dfs(,);
for (RE LL i=;i<n;i++)
{
LL u=Rank[i];
while (sum[u]>m)
{
sum[u]-=root[u]->cost;
remain[u]--;
root[u]=Merge(root[u]->l,root[u]->r);
}
sum[father[u]]+=sum[u];
remain[father[u]]+=remain[u];
root[father[u]]=Merge(root[father[u]],root[u]);
}
LL ans=-INF;
for (RE LL i=;i<=n;i++) ans=Max(ans,H[i].key*remain[i]);
printf("%lld\n",ans);
return ;
} IL void Add(LL u,LL v)
{
edge[++top].to=v;
edge[top].next=path[u];
path[u]=top;
}
void Dfs(LL r,LL father)
{
for (RE LL i=path[r];i;i=edge[i].next) if (edge[i].to!=father) Dfs(edge[i].to,r);
Rank[tail++]=r;
}
node* Merge(node* a,node* b)
{
if (!a||!b) return a ? a:b;
if (a->cost<b->cost) swap(a,b);
a->r=Merge(a->r,b);
if (a->ldist()<a->rdist()) swap(a->l,a->r);
a->dist=a->rdist()+;
return a;
}

[APIO 2012]派遣的更多相关文章

  1. APIO 2012 派遣(可并堆)

    APIO 2012 派遣(可并堆) 给定一棵N个点的树和M,每个点有两个权值ai,bi,每次可以选择一个点x,然后在这个点的子树中选若干点(可以不选自己),使得这些点的\(\sum b_i<=M ...

  2. 解题:APIO 2012 派遣

    题面 以报酬为标准维护一个大根堆,从根节点往上合并,每次踢掉若干人直到花费合法后更新答案 #include<cstdio> #include<cstring> #include ...

  3. 【BZOJ 2809】【APIO 2012】dispatching

    昨天晚上zyf神犇问我的题,虽然我太弱参加不了APIO但也做一做吧. 用小数据拍了无数次总是查不出错来,交上去就WA,后来用国内数据测发现是主席树上区间相减的值没有用long long存,小数据真是没 ...

  4. 「BZOJ 2809」「APIO 2012」Dispatching「启发式合并」

    题意 给定一个\(1\)为根的树,每个点有\(c,w\)两个属性,你需要从某个点\(u\)子树里选择\(k\)个点,满足选出来的点\(\sum_{i=1}^k w(i)\leq m\),最大化\(k\ ...

  5. BZOJ 2809 APIO 2012 dispatching 平衡树启示式合并

    题目大意:给出一棵树,每个节点有两个值,各自是这个忍者的薪水和忍者的领导力.客户的惬意程度是这个点的领导力乘可以取得人数.前提是取的人的薪水总和不超过总的钱数. 思路:仅仅能在子树中操作.贪心的想,我 ...

  6. SYSU 6356 Dispatching

    Dispatching Time Limit: 3000ms Memory Limit: 262144KB This problem will be judged on SYSU. Original ...

  7. 2012 A 《中国近现代史纲要》课程期末考试试卷

    湖南人文科技学院2012—2013学年第1学期公共课 2011级<中国近现代史纲要>课程期末考试试卷 考核方式:(开卷)                                   ...

  8. 2012年5月阿里巴巴集团”去 IOE”运动的思考与总结【转载+整理】

    原文地址 什么是 IOE,IOE 只是一个简称,分别代表 IBM.Oracle.EMC,确切地说是 IBM 小型机.Oracle 数据库与 EMC 存储设备的组合.这"三驾马车"构 ...

  9. Windows server 2012 添加中文语言包(英文转为中文)(离线)

    Windows server 2012 添加中文语言包(英文转为中文)(离线) 相关资料: 公司环境:亚马孙aws虚拟机 英文版Windows2012 中文SQL Server2012安装包,需要安装 ...

随机推荐

  1. hackme.inndy.tw - pyyy - Writeup

    hackme.inndy.tw - pyyy - Writeup 0x01 反编译 1.第一次尝试的时候我直接在线反编译,部分结果如下. for (i, f) in enumerate(F): n = ...

  2. 【福大软工】 W班级总成绩排名2

    评分链接: 选题报告    结对第二次作业     需求分析    随堂测试 总分排名:

  3. 敏捷冲刺每日报告二(Java-Team)

    第二天报告(10.26  周四) 团队:Java-Team 成员: 章辉宇(284) 吴政楠(286) 陈阳(PM:288) 韩华颂(142) 胡志权(143) github地址:https://gi ...

  4. C语言——第0次作业(二)

    1.翻阅邹欣老师的关于师生关系博客,并回答下列问题,每个问题的答案不少于500字: 1.最理想的师生关系是健身教练和学员的关系,在这种师生关系中你期望获得来自老师的哪些帮助? 在现代软件工程讲义 0 ...

  5. iOS开发-即时通信XMPP

    1. 即时通信 1> 概述 即时通讯(Instant Messaging)是目前Internet上最为流行的通讯方式,各种各样的即时通讯软件也层出不穷,服务提供商也提供了越来越丰富的通讯服务功能 ...

  6. Flask 学习 十六 部署

    部署流程 manage.py 部署命令 每次安装升级只需运行deploy命令即可完成操作 @manager.command def deploy(): """执行部署任务 ...

  7. mysql常用命令整理

    #不压缩备份 mysqldump -u root -p userpassword databasename > /tmp/backupfile.sql #压缩备份 mysqldump -u ro ...

  8. C# 启动 SQL Server 服务

    //首先要添加 System.ServiceProcess.dll 引用 ServiceController sc = new ServiceController("MSSQLSERVER& ...

  9. (干货)微信小程序之上传图片和图片预览

    这几天一直负责做微信小程序这一块,也可以说是边做边学习吧,把自己做的微信小程序的一些功能分享出来,与大家探讨一下,相互学习相互进步. 先看下效果图 只写了一下效果样式的话希望大家不要太在意,下面马路杀 ...

  10. split 过滤空的元素

    命令形式: split(str='',number=string.count(str))[n] str 分隔符 number 切分几次,[n] 获取第几个值. 1.如果切分的可迭代对象中包含空元素的解 ...