Codeforces 题目传送门 & 洛谷题目传送门

一道不算太难的贪心,可惜又没自己想出来,显然省选之后我的能力呈 \(y=-1145141919810192608179998244353x+c\) 的趋势下滑,其中 \(c\) 为我省选前的能力(

首先假设我们已经选出了这 \(k\) 个操作,考虑按照怎样的顺序执行这些操作,显然我们会先执行赋值操作,再加法操作,最后乘法操作,因为赋值操作早晚都是要进行的,按照贪心的思想显然在一开始就进行赋值操作最优,而对于序列中某个元素 \(x\),我们假设它需进行 \(+y,\times z\) 两个操作,那么先加后乘的结果为 \(z(x+y)\),先乘后加的结果为 \(xz+y\),做差可得 \(\Delta=y(z-1)\ge 0\),故先加后乘必然比先乘后加来得更优。

接下来考虑怎样选出这些操作。首先很明显对于一个 \(\times z\) 的操作,不管它作用在哪个数上,都会导致最后的结果也乘 \(z\),也就是说如果只有乘法操作的话,我们可以直接将 \(z\) 从大到小排序贪心。那加上加法和赋值操作该怎么办呢?有一件很显然的事情是赋值操作可以转化为加法操作,因为显然每个数最多被赋一次值,并且给下标为 \(i\) 的元素赋的值只可能是所有作用在 \(i\) 的赋值操作中值最大的,假设为 \(x_i\),那么我们就可以将作用在 \(i\) 上的赋值操作看作一次 \(+(x_i-a_i)\) 的加法操作。这样就只有加法操作了,我们还可以发现,对于加法操作我们肯定会按加的值从大到小选择,也就是说对于一次作用在 \(i\) 上的加法操作 \(+v\),假如我们执行了这次操作,那么在此次操作前 \(a_i\) 的值必然是一个定值,也就是说我们也可以将这次操作看作 \(\times\dfrac{a_i+v}{a_i}\)。这样所有加法、赋值操作都可以转化为乘法操作,也就可以按照最一开始的贪心策略贪了,时间复杂度 \(n\log n\)。

这里有一个小的注意点,就是我们由加法转化来的乘法操作会出现分数,此时我们就要对分数比较大小,而这题分数的分子分母会达到 \(10^5\times 10^6=10^{11}\),直接比较大小运算量会达到 \(10^{11}\times 10^{11}=10^{22}\),爆 ll,如果我没记错的话这样写大概会 WA 146,这里有一个小技巧,就是所有分数值都减去 \(1\),这样 \(\times\dfrac{a_i+v}{a_i}\) 就变成了 \(\dfrac{v}{a_i}\),分子大小就降到了 \(10^{6}\),直接排就能过了。

const int MAXN=1e5;
int n,m,k,ini[MAXN+5],c1,c2,c3;
struct event{int opt,id,x;ll a,b;} q1[MAXN+5],q2[MAXN+5],q3[MAXN+5];
bool cmp1(event lhs,event rhs){return (lhs.x^rhs.x)?(lhs.x<rhs.x):(lhs.a<rhs.a);}
bool cmp2(event lhs,event rhs){return (lhs.x^rhs.x)?(lhs.x<rhs.x):(lhs.a>rhs.a);}
bool cmp3(event lhs,event rhs){return lhs.a*rhs.b>rhs.a*lhs.b;}
bool cmp4(event lhs,event rhs){return lhs.opt<rhs.opt;}
int main(){
scanf("%d%d%d",&n,&m,&k);
for(int i=1;i<=n;i++) scanf("%d",&ini[i]);
for(int i=1;i<=m;i++){
int opt,x,y;scanf("%d%d%d",&opt,&x,&y);
if(opt==1&&y>ini[x]) q1[++c1]={1,i,x,y,1};
else if(opt==2) q2[++c2]={2,i,x,y,1};
else if(opt==3) q3[++c3]={3,i,x,y,1};
} sort(q1+1,q1+c1+1,cmp1);
for(int i=1;i<=c1;i++) if(q1[i].x!=q1[i+1].x)
q2[++c2]={q1[i].opt,q1[i].id,q1[i].x,q1[i].a-ini[q1[i].x],1};
sort(q2+1,q2+c2+1,cmp2);ll sum=0;
for(int i=1;i<=c2;i++){
if(q2[i].x!=q2[i-1].x) sum=ini[q2[i].x];
q3[++c3]={q2[i].opt,q2[i].id,q2[i].x,sum+q2[i].a,sum};sum+=q2[i].a;
// printf("%d %d %d %lld\n",q2[i].opt,q2[i].id,q2[i].x,q2[i].a);
} for(int i=1;i<=c3;i++) q3[i].a-=q3[i].b;
sort(q3+1,q3+c3+1,cmp3);k=min(c3,k);
sort(q3+1,q3+k+1,cmp4);printf("%d\n",k);
for(int i=1;i<=k;i++) printf("%d ",q3[i].id);
return 0;
}

Codeforces 521D - Shop(贪心)的更多相关文章

  1. CodeForces - 158B.Taxi (贪心)

    CodeForces - 158B.Taxi (贪心) 题意分析 首先对1234的个数分别统计,4人组的直接加上即可.然后让1和3成对处理,只有2种情况,第一种是1多,就让剩下的1和2组队处理,另外一 ...

  2. codeforces 724D(贪心)

    题目链接:http://codeforces.com/contest/724/problem/D 题意:给定一个字符串和一个数字m,选取一个一个子序列s,使得对于字符串中任意长度为m的子序列都至少含有 ...

  3. Codeforces 626G Raffles(贪心+线段树)

    G. Raffles time limit per test:5 seconds memory limit per test:256 megabytes input:standard input ou ...

  4. Cut 'em all! CodeForces - 982C(贪心dfs)

    K - Cut 'em all! CodeForces - 982C 给一棵树 求最多能切几条边使剩下的子树都有偶数个节点 如果n是奇数 那么奇数=偶数+奇数 不管怎么切 都会有奇数 直接打印-1 贪 ...

  5. CodeForces - 940E - Cashback +贪心+DP

    传送门:CodeForces - 940E - Cashback 题意:在一个长度为n的数组中,可以分出长度为 k 连续的多个数组b(每个数组 b 的 k 可不相同),然后,可以对每个数组 b 进行删 ...

  6. Codeforces 515C 题解(贪心+数论)(思维题)

    题面 传送门:http://codeforces.com/problemset/problem/515/C Drazil is playing a math game with Varda. Let’ ...

  7. CodeForces 485C Bits[贪心 二进制]

    C. Bits time limit per test1 second memory limit per test256 megabytes inputstandard input outputsta ...

  8. codeforces 732E(贪心)

    题目链接:http://codeforces.com/contest/732/problem/E 题意:有n台计算机,m个插座,每台计算机有一个值a[i],每个插座有一个值b[i],每个插座最多只能对 ...

  9. Codeforces 732D [二分 ][贪心]

    /* 不要低头,不要放弃,不要气馁,不要慌张 题意: n天进行m科考试,每科考试需要a的复习时间,n天每天最多可以考一科.并且指定哪天考哪科. 注意考试那天不能复习. 问最少需要多少天可全部通过考试. ...

随机推荐

  1. 【UE4】GAMES101 图形学作业1:mvp 模型、视图、投影变换

    总览 到目前为止,我们已经学习了如何使用矩阵变换来排列二维或三维空间中的对象.所以现在是时候通过实现一些简单的变换矩阵来获得一些实际经验了.在接下来的三次作业中,我们将要求你去模拟一个基于CPU 的光 ...

  2. 第二次Scrum Metting

    日期:2021年4月25日会议主要内容概述:前后端针对WebAPI进行协调与统一工作,商量接下来两日计划:敲定部分设计细节. 一.进度情况 组员 负责 两日内已完成的工作 后两日计划完成的工作 工作中 ...

  3. 一张图彻底搞懂Spring循环依赖

    1 什么是循环依赖? 如下图所示: BeanA类依赖了BeanB类,同时BeanB类又依赖了BeanA类.这种依赖关系形成了一个闭环,我们把这种依赖关系就称之为循环依赖.同理,再如下图的情况: 上图中 ...

  4. 加法运算替代 牛客网 程序员面试金典 C++ Python

    加法运算替代 牛客网 程序员面试金典 题目描述 请编写一个方法,实现整数的乘法.减法和除法运算(这里的除指整除).只允许使用加号. 给定两个正整数int a,int b,同时给定一个int type代 ...

  5. hdu 5166 Missing number(。。。)

    题意: 有一个排列,但少了两个数.给你少了这两个数的排列.找出是哪两个数. 思路: 看代码,,, 代码: int a[1005]; int main(){ int T; cin>>T; w ...

  6. hdu 1160 FatMouse's Speed(最长不下降子序列+输出路径)

    题意: FatMouse believes that the fatter a mouse is, the faster it runs. To disprove this, you want to ...

  7. 微信小程序API接口封装

    @ 目录 一,让我们看一下项目目录 二,让我们熟悉一下这三个文件目的(文件名你看着办) 三,页面js中如何使用 今天的API的封装,我们拿WX小程序开发中,对它的API (wx.request)对这个 ...

  8. spring mvc 原理(快速理解篇)

    这两张图大家应该都不陌生. 从图上来看就是:一个请求过来,front controller根据具体的请求路径分派到具体的controller,具体的controller处理请求并把处理结果返回给fro ...

  9. Docker+nginx搭建tomcat集群

    1.环境准备: a.宿主机CentOS7 b.连接工具FinalShell c.镜像nginx1.20.1,tomcat (镜像拉取:docker pull 镜像名称) 2.创建nginx文件夹,to ...

  10. Spark记录(一):Spark全景概述

    一.Spark是什么 Spark是一个开源的大数据处理引擎. 二.Spark的主要组件如下图所示:  三.Spark运行时架构 Spark共有三种运行模式:本地模式.集群模式.客户端模式. 生产环境基 ...