洛谷题面传送门 & Atcoder 题面传送门

由于再过 1h 就是 NOI 笔试了所以题解写得会略有点简略。

考虑差分,记 \(b_i=c_i-c_{fa_i}\),那么根据题意有 \(b_i\le d,i=2,3,4,\cdots,n\),而 \(b_1\) 则没有任何约束条件。而如果我们令某个 \(b_i\) 加 \(1\),其余 \(b_i\) 均不变,那对应到原 \(c\) 序列上的操作效果就是 \(i\) 子树内的 \(c_j\) 加 \(1\),其余 \(c_j\) 不变,因此预处理出 \(i\) 子树大小 \(siz_i\) 以及 \(i\) 子树内所有 \(m_j\) 之和 \(w_i\),问题就转化为:有 \(n\) 类物品,第 \(i\) 类代价为 \(w_i\),价值为 \(siz_i\),第 \(2,3,4,\cdots,n\) 类物品有 \(d\) 个,第 \(1\) 类物品有无限个,求用不超过 \(X\) 的代价买到的物品的最大价值和。

一眼多重背包,不过此题数据范围高达 \(10^9\),直接稳了(指稳 TLE+MLE)。不过考虑一个错误的贪心,将所有物品按性价比 \(\dfrac{w_i}{siz_i}\) 从小到大排序然后贪心地取,每次能多取就多取,不过这个贪心显然是假的,反例随便举。不过这个贪心给了我们一点启发:对于两个满足 \(\dfrac{w_i}{siz_i}<\dfrac{w_j}{siz_j}\) 的物品 \(i,j\),如果目前 \(i\) 还能够再多取 \(siz_j\) 个,并且 \(j\) 目前取的个数多余 \(siz_i\) 个,那么我们完全可以让 \(j\) 少取 \(siz_i\) 个,\(i\) 多取 \(siz_j\) 个,这样肯定比原方案更优,也就是说我们钦定每个物品前 \(n\) 个中取了多少个,钦定完之后肯定就优先取 \(\dfrac{w_i}{siz_i}\) 小的物品更优了。

因此考虑将所有物品先拿 \(\min(n,d)\) 个出来多重背包,即记 \(f_i\) 表示最少需要多少的代价才能够获得 \(i\) 的收益,剩余部分贪心即可,多重背包可以通过二进制分组/单调队列优化,复杂度 \(\mathcal O(n^4\log n)/\mathcal O(n^4)\),本人使用的二进制分组。

const int MAXN=50;
const int MAXV=1.25e5;
const int MAXI=350;
int n,m,d,siz[MAXN+5],ord[MAXN+5];ll w[MAXN+5];
int hd[MAXN+5],to[MAXN+5],nxt[MAXN+5],ec=0;
void adde(int u,int v){to[++ec]=v;nxt[ec]=hd[u];hd[u]=ec;}
void dfs(int x){
siz[x]=1;
for(int e=hd[x];e;e=nxt[e]){
int y=to[e];dfs(y);
siz[x]+=siz[y];w[x]+=w[y];
}
}
bool cmp(int x,int y){return 1ll*w[x]*siz[y]<1ll*w[y]*siz[x];}
ll dp[MAXV+5],W[MAXI+5];
int V[MAXI+5],item_n=0,lim[MAXN+5];
int main(){
scanf("%d%d%d%lld",&n,&m,&d,&w[1]);
for(int i=2,f;i<=n;i++) scanf("%lld%d",&w[i],&f),adde(f,i);
dfs(1);for(int i=1;i<=n;i++) ord[i]=i;sort(ord+1,ord+n+1,cmp);
int mx=0;memset(dp,63,sizeof(dp));dp[0]=0;
for(int i=1;i<=n;i++){
int up=min((ord[i]==1)?0x3f3f3f3f:d,n);
mx+=up*siz[ord[i]];
int k=31-__builtin_clz(up);
for(int j=0;j<k;j++){
W[++item_n]=w[ord[i]]<<j;
V[item_n]=siz[ord[i]]<<j;
} W[++item_n]=w[ord[i]]*(up-(1<<k)+1);
V[item_n]=siz[ord[i]]*(up-(1<<k)+1);
lim[ord[i]]=((ord[i]==1)?0x3f3f3f3f:d)-up;
// printf("%d %d\n",ord[i],lim[ord[i]]);
} int ans=0;
for(int i=1;i<=item_n;i++) for(int j=mx;j>=V[i];j--)
chkmin(dp[j],dp[j-V[i]]+W[i]);
for(int i=0;i<=mx;i++) if(dp[i]<=m){
int lft=m-dp[i],sum=i;//printf("%d %d %d\n",i,lft,sum);
for(int j=1;j<=n;j++){
int can=min((int)(lft/w[ord[j]]),lim[ord[j]]);
// printf("%d\n",can);
sum+=can*siz[ord[j]];lft-=can*w[ord[j]];
} chkmax(ans,sum);
} printf("%d\n",ans);
return 0;
}

Atcoder Regular Contest 096 D - Sweet Alchemy(贪心+多重背包)的更多相关文章

  1. AtCoder Regular Contest 096

    AtCoder Regular Contest 096 C - Many Medians 题意: 有A,B两种匹萨和三种购买方案,买一个A,买一个B,买半个A和半个B,花费分别为a,b,c. 求买X个 ...

  2. [AtCoder Regular Contest 096 E] Everything on It 解题报告 (第二类斯特林数+容斥原理)

    题目链接:https://arc096.contest.atcoder.jp/tasks/arc096_c Time limit : 4sec / Memory limit : 512MB Score ...

  3. Atcoder Regular Contest 096 C - Everything on It(组合数学)

    Atcoder 题面传送门 & 洛谷题面传送门 简单题,由于这场 arc 的 F 是 jxd 作业而我不会做,所以只好来把这场的 E 水掉了. 我们记 \(f(i)\) 为钦定 \(i\) 个 ...

  4. AtCoder Regular Contest 096 D - Static Sushi(线性dp)

    Problem Statement "Teishi-zushi", a Japanese restaurant, is a plain restaurant with only o ...

  5. AtCoder Regular Contest 098

    AtCoder Regular Contest 098 C - Attention 题意 给定一个只包含"E","W"字符串,可以花一的花费使他们互相转换.选定 ...

  6. AtCoder Regular Contest 061

    AtCoder Regular Contest 061 C.Many Formulas 题意 给长度不超过\(10\)且由\(0\)到\(9\)数字组成的串S. 可以在两数字间放\(+\)号. 求所有 ...

  7. AtCoder Regular Contest 094 (ARC094) CDE题解

    原文链接http://www.cnblogs.com/zhouzhendong/p/8735114.html $AtCoder\ Regular\ Contest\ 094(ARC094)\ CDE$ ...

  8. AtCoder Regular Contest 092

    AtCoder Regular Contest 092 C - 2D Plane 2N Points 题意: 二维平面上给了\(2N\)个点,其中\(N\)个是\(A\)类点,\(N\)个是\(B\) ...

  9. AtCoder Regular Contest 093

    AtCoder Regular Contest 093 C - Traveling Plan 题意: 给定n个点,求出删去i号点时,按顺序从起点到一号点走到n号点最后回到起点所走的路程是多少. \(n ...

随机推荐

  1. 小白自制Linux开发板 八. Linux音频驱动配置

    不知不觉小白自制开发板系列已经到第八篇了,本篇要配置的是音频驱动,也算是硬件部分的最后一片了,积攒的文章也差不多全放完了,后续更新可能会放缓,还请见谅. 对于F1C200s是自带了多媒体处理功能的,所 ...

  2. JDK 8中重要的函数式接口(必知必会)

    JDK 8 提供的重要函数式接口: Consumer (消费者) 功能:接收一个对象,返回void. 定义:void accept(T t) 默认方法:Consumer andThen(Consume ...

  3. (数据科学学习手札129)geopandas 0.10版本重要新特性一览

    本文示例代码及文件已上传至我的Github仓库https://github.com/CNFeffery/DataScienceStudyNotes 1 简介 就在前不久,我们非常熟悉的Python地理 ...

  4. Noip模拟35 2021.8.10

    考试题目变成四道了,貌似确实根本改不完... 不过给了两个小时颓废时间确实很爽(芜湖--) 但是前几天三道题改着不是很费劲的时候为什么不给放松时间, 非要在改不完题的时候颓?? 算了算了不碎碎念了.. ...

  5. [BZOI2014]大融合——————线段树进阶

    竟然改了不到一小时就改出来了, 可喜可贺 Description Solution 一开始想的是边两侧简单路径之和的乘积,之后发现这是个树形结构,简单路径数就是节点数. 之后的难点就变成了如何求线段树 ...

  6. stat命令的实现

    任务详情 学习使用stat(1),并用C语言实现 提交学习stat(1)的截图 man -k ,grep -r的使用 伪代码 产品代码 mystate.c,提交码云链接 测试代码,mystat 与st ...

  7. Linux ps -ef 命令输出解释

    UID: 程序拥有者PID:程序的 IDPPID:程序父级程序的 IDC:  CPU 使用的百分比STIME: 程序的启动时间TTY: 登录终端TIME : 程序使用掉 CPU 的时间CMD: 下达的 ...

  8. ffmpeg第7篇:数据流选择神器-map指令

    自动选择规则 ffmpeg在处理视频时,如果只提供了输入和输出参数,ffmpeg会自动地去选择相应的视频流和音频流来合成文件 自动选择的方式根据如下规则: 视频流:选分辨率最高的,比如有两个视频,一个 ...

  9. 记一次CTF比赛过程与解题思路-MISC部分

    前言 最近好久没更新博客和公众号了,有朋友问是不是在憋大招,但我不好意思说其实是因为最近一段时间太懒了,一直在当咸鱼- 意识到很久没更新这个问题,我是想写点什么的,但好像一直当咸鱼也没啥可分享的,最近 ...

  10. Linux部署Apollo+.Net Core简单使用

    Apollo官方网站非常详细,以下只是本人学习过程的整理 一.概念 Apollo(阿波罗)是一款可靠的分布式配置管理中心,能够集中化管理应用不同环境.不同集群的配置,配置修改后能够实时推送到应用端,并 ...