Atcoder Regular Contest 125 E - Snack(最小割转化+贪心)
Preface:
这是生平第一道现场 AC 的 arc E,也生平第一次经历了 performance \(\ge 2800\),甚至还生平第一次被 hb 拉到会议里讲题,讲的就是这个题,然鹅比较尬的一点是我不知道腾讯会议开了白板之后不能看到电脑,导致我的 typora 没人能看到……所以就暂且把我 typora 的内容整了整改成了一篇题解(
题意:
- 有 \(n\) 种零食,第 \(i\) 种零食有 \(a_i\) 个。
- 有 \(m\) 个人领这些零食,第 \(i\) 个人最多领 \(b_i\) 个同种零食
- 第 \(i\) 个人领零食的总数不能超过 \(c_i\)
- 求最多能分出去多少零食
- \(n,m\le 2\times 10^5,a_i\le 10^{12},b_i\le 10^7\)
首先考虑网络流建图,我们新建一个源点 \(S\) 和汇点 \(T\),然后考虑将零食看作左部点 \(X_i\),人看作右部点 \(Y_i\),那么我们显然可以建出以下几类边:
- 对每个 \(i\in[1,n]\) 连边 \(S\to X_i\),容量 \(a_i\)
- 对每个 \(i\in[1,n],j\in[1,m]\) 连边 \(X_i\to Y_j\),容量 \(b_j\)
- 对每个 \(i\in[1,m]\) 连边 \(Y_i\to T\),容量 \(c_i\)
然后跑最大流就是答案。不过显然直接建图复杂度爆炸——甚至连图都存不下,因此考虑这样一个套路:使用最大流&最小割定理转化为最小割问题。也就是我们要割掉权值最小的边集使得 \(S,T\) 不连通。下记 \(S\to X_i\) 为第一类边,\(X_i\to Y_j,Y_i\to T\) 分别记作第二、三类边。
这个问题就可以贪心求解了。考虑枚举保留多少个一类边,记作 \(x\),那注意到此题的一个性质,所有左部点(也就是 \(X_i\))在我们钦定割/不割掉与 \(S\) 相连的边后就是等价的了,也就是说只有切掉的个数有意义,具体切的是哪些边对后面(也就是切二、三类边的代价)的计算没有影响。因此我们肯定会贪心地割掉最小的 \(n-x\) 条 一类边,排序+前缀和求解即可。接下来考虑割掉二、三类边的代价,对于一个 \(Y_j\),目前有用的边肯定只剩下 \(x\) 条与 \(Y_j\) 相连的二类边和 \(Y_j\) 连向汇点的三类边,显然要么二类边全删,要么三类边全删,因此删掉与 \(j\) 有关的边的代价是 \(\min(c_j,b_j·x)\)。将所有右部点按照 \(\dfrac{c_j}{b_j}\) 排序后 \(\min\) 取到 \(c_j\) 的部分一定是一段前缀,\(\min\) 取到 \(b_j·x\) 的部分肯定是对应部分的一个补集,也就是一段后缀。我们在枚举 \(x\) 的过程中 two pointers 找出这段前缀,然后预处理出 \(b,c\) 的前缀和即可 \(\mathcal O(1)\) 计算贡献。
时间复杂度 \(n\log n\)(如果 \(n,m\) 视作同阶),注意精度,直接比较 double 可能会获得 \(\text{AC}\times26,\text{WA}\times1\) 的好成绩。
const int MAXN=2e5;
int n,m,ord[MAXN+5];
ll a[MAXN+5],b[MAXN+5],c[MAXN+5];
ll suma[MAXN+5],sumb[MAXN+5],sumc[MAXN+5];
bool cmp(int x,int y){return 1ull*c[x]*b[y]<1ull*b[x]*c[y];}
int main(){
scanf("%d%d",&n,&m);
for(int i=1;i<=n;i++) scanf("%lld",&a[i]);
for(int i=1;i<=m;i++) scanf("%lld",&b[i]);
for(int i=1;i<=m;i++) scanf("%lld",&c[i]);
for(int i=1;i<=m;i++) ord[i]=i;
sort(a+1,a+n+1);sort(ord+1,ord+m+1,cmp);
for(int i=1;i<=n;i++) suma[i]=suma[i-1]+a[i];
for(int i=1;i<=m;i++) sumb[i]=sumb[i-1]+b[ord[i]];
for(int i=1;i<=m;i++) sumc[i]=sumc[i-1]+c[ord[i]];
ll res=1e18;
for(int i=0;i<=n;i++){
int l=1,r=m,p=0;
while(l<=r){
int mid=l+r>>1;
if(1ll*b[ord[mid]]*i<=c[ord[mid]]) r=mid-1;
else p=mid,l=mid+1;
} //printf("%d %d\n",i,p);
chkmin(res,suma[n-i]+sumc[p]+1ll*(sumb[m]-sumb[p])*i);
} printf("%lld\n",res);
return 0;
}
Atcoder Regular Contest 125 E - Snack(最小割转化+贪心)的更多相关文章
- AtCoder Regular Contest 061
AtCoder Regular Contest 061 C.Many Formulas 题意 给长度不超过\(10\)且由\(0\)到\(9\)数字组成的串S. 可以在两数字间放\(+\)号. 求所有 ...
- AtCoder Regular Contest 093
AtCoder Regular Contest 093 C - Traveling Plan 题意: 给定n个点,求出删去i号点时,按顺序从起点到一号点走到n号点最后回到起点所走的路程是多少. \(n ...
- AtCoder Regular Contest 094
AtCoder Regular Contest 094 C - Same Integers 题意: 给定\(a,b,c\)三个数,可以进行两个操作:1.把一个数+2:2.把任意两个数+1.求最少需要几 ...
- AtCoder Regular Contest 095
AtCoder Regular Contest 095 C - Many Medians 题意: 给出n个数,求出去掉第i个数之后所有数的中位数,保证n是偶数. \(n\le 200000\) 分析: ...
- AtCoder Regular Contest 102
AtCoder Regular Contest 102 C - Triangular Relationship 题意: 给出n,k求有多少个不大于n的三元组,使其中两两数字的和都是k的倍数,数字可以重 ...
- AtCoder Regular Contest 096
AtCoder Regular Contest 096 C - Many Medians 题意: 有A,B两种匹萨和三种购买方案,买一个A,买一个B,买半个A和半个B,花费分别为a,b,c. 求买X个 ...
- AtCoder Regular Contest 098
AtCoder Regular Contest 098 C - Attention 题意 给定一个只包含"E","W"字符串,可以花一的花费使他们互相转换.选定 ...
- AtCoder Regular Contest 099
AtCoder Regular Contest 099 C - Minimization 题意 题意:给出一个n的排列.每次操作可以使一段长度为K的连续子序列变成该序列的最小数.求最少几次使得整个数列 ...
- AtCoder Regular Contest 069 F Flags 二分,2-sat,线段树优化建图
AtCoder Regular Contest 069 F Flags 二分,2-sat,线段树优化建图 链接 AtCoder 大意 在数轴上放上n个点,点i可能的位置有\(x_i\)或者\(y_i\ ...
随机推荐
- Sequence Model-week2编程题2-Emoji表情生成器
1. Emoji表情生成器 下面,我们要使用词向量(word vector)来构建一个表情生成器. 你将实现一个模型:输入一句话 (如 "Let's go see the baseball ...
- SDIO总线协议
SDIO采用HOST-DEVICE模式,所有通信都由HOST端发命令,DEVICE设备只要解析HOST命令就可与HOST进行通信. SDIO总线的几根线: 1. CLK信号:HOST给DEVICE的 ...
- 从0到1使用Kubernetes系列(五):Kubernetes Scheduling
前述文章介绍了Kubernetes基本介绍,搭建Kubernetes集群所需要的工具,如何安装,如何搭建应用.本篇介绍怎么使用Kubernetes进行资源调度. Kubernetes作为一个容器编排调 ...
- gcc中预定义的宏__GNUC__
转载:gcc中预定义的宏__GNUC__ - Cccarl - 博客园 (cnblogs.com) 今天在看Linux系统编程这本书的代码的时候看到了__GNUC__,不太清楚这个宏所以去查了一下,以 ...
- 树形DP 枚举祖宗的例题
这类题目是真的很头疼....其实这类题目的特征也很明显,叶子结点贡献答案时和其所在链的祖宗有关,也就是说要想得知其贡献必须知道他的所有祖宗的贡献,其实处理方法也不是太难,就是在dfs枚举时顺便把祖宗的 ...
- cf13C Sequence(DP)
题意: N个数.a1...aN. 对于每个数而言,每一步只能加一或减一. 问最少总共需要多少步使得新序列是非递减序列. N (1 ≤ N ≤ 5000) 思路: *一个还不知道怎么证明的结论(待证): ...
- Vue-cli4.xPC端项目Rem适配
适配准备 安装 (amfe-flexible) 和(postcss-px2rem) 1, 安装依赖并在main.js中引入该依赖 npm i amfe-flexible import "am ...
- mongoDB 的一般使用
理解 mongodb 也是nosql 的一种.他的数据存储类型是一种和json格式比较像的数据类型,可以看作就是json. mongodb 里的数据库都是一个单独的库.一般需要用的库都会设置自己的us ...
- LeetCode刷题 树专题
树专题 关于树的几个基本概念 1 树的节点定义 2 关于二叉树的遍历方法 2.1 前序遍历 2.2 中序遍历 2.3 后序遍历 2.4 层序遍历 3 几种常见的树介绍 3.1 完全二叉树 3.2 二叉 ...
- celery config
/* Useful celery config. app = Celery('tasks', broker='redis://localhost:6379', backend='redis://loc ...