题解 洛谷P3745 【[六省联考2017]期末考试】
这题有点绕,我写了\(2h\)终于搞明白了。
主要思路:枚举最晚公布成绩的时间\(maxt\),然后将所有公布时间大于\(maxt\)的课程都严格降为\(maxt\)即可。
在此之前,还要搞清楚一个概念:对于第二种操作,它只有将某一门课提前,但是第一种操作,它还会在提前的过程中延迟某一门课。所以,在不考虑代价的情况下,选择第二种操作是更优也更快捷的。
接下来考虑分情况贪心。
如果\(A \geq B\),说明第一种操作比第二种操作的代价来的高,操作也没有第二种优,所以肯定优先选择第二种操作。
如果\(A<B\),说明第一种操作的代价比较小,所以可以先选第一种到不能选为止(已经没有课程可以延迟),剩余的再选第二种。
于是再算一下在最晚公布时间为\(i\)的情况下学生们的不愉快度即可。
实现过程
- \(A \geq B\)的情况
此时我们要计算有多少门课公布的时间大于\(i\),我们可以先考虑将所有的课程按公布时间排序,然后把公布时间比\(i\)晚的课程先计算出来,这里我就用\(x\)来表示数量,则将这些课程公布的时间记作\(a_{m-x+1},a_{m-x+2}……a_m\)。
而我们的意图是将这里所有的值都化为\(i\),这其中经过的时间为\(a_{m-x+1}+a_{m-x+2}+……+a_m-i \times x\)
代价为\((a_{m-x+1}+a_{m-x+2}+……+a_m-i \times x) \times B\)
这个东西我们可以用前缀和\(O(1)\)的求出,而\(x\)的值怎么求可以参考我的代码。
- \(A<B\)的情况
一样的,先算出要将其化为\(i\)需要经过的时间:
\(a_{m-x+1}+a_{m-x+2}+……+a_m-i \times x\) \((\)记作\(t1)\)
然后再算比\(i\)早公布时间的课程,我们可以将其延迟至\(i\)时间公布。
\(i \times (m-x) -(a_{1}+a_{2}+……+a_x)\) \((\)记作\(t2)\)
比较\(t1\)和\(t2\),如果\(t1 \leq t2\),说明可以全部用第一种操作搞定,代价为:\(t1 \times A\)
否则,剩余的再用第二种操作,代价为:\(t2 \times A+(t1-t2) \times B\)
- 学生的不愉快度
考虑用一个\(Num\)记录在最晚公布时间为\(i\)的情况下不愉快的学生,将\(t_i\)从小到大排序,则不愉快的学生的等待时间记为:\(t_1,t_2,t_3……t_{Num}\)
总等待时间为:
\(i \times Num-(t_1+t_2+t_3……+t_{Num})\)
不愉快度为:
\((i \times Num-(t_1+t_2+t_3……+t_{Num})) \times C\)
那么最后注意一下用\(ull\),然后套柿子就没了。
\(Code:\)
#include<bits/stdc++.h>
#define long unsigned long long
using namespace std;
inline int read(){
register int s=0,f=1;
register char ch=getchar();
while(!isdigit(ch)){if(ch=='-')f*=-1;ch=getchar();}
while(isdigit(ch))s=(s<<1)+(s<<3)+(ch^48),ch=getchar();
return s*f;
}
const int max_n=100000+5;
long late[max_n],Time[max_n];
long sum[max_n],f[max_n],num[max_n],h[max_n],maxt;
long Max(long a,long b){
if(a>b)return a;
return b;
}
long Min(long a,long b){
if(a<b)return a;
return b;
}
int main(){
ios::sync_with_stdio(false);
long A=read(),B=read(),C=read();
long n=read(),m=read();
for(int i=1;i<=n;i++)late[i]=read();
for(int i=1;i<=m;i++)Time[i]=read();//读入
sort(Time+1,Time+m+1);//排序
for(int i=1;i<=m;i++){
maxt=max(maxt,Time[i]),sum[Time[i]]++;
//找最晚公布的时间 //记录当前值,方便统计
f[i]=f[i-1]+Time[i];//求前缀和
}
sort(late+1,late+n+1);//排序
long Num=0;//不愉快的学生
for(int i=1;i<=n;i++){
num[late[i]]++,h[i]=h[i-1]+late[i];
if(late[i]<=maxt)Num=i;
}
long ans=LONG_LONG_MAX;
long x=0;//公布时间比i晚的课程
for(int i=maxt;i>=1;i--){
Num-=num[i];
long tot=0;
if(A>=B){
long day=f[m]-f[m-x]-x*i;
tot=Max(day*B,0);
}else{
long day1=f[m]-f[m-x]-x*i;
long day2=(m-x)*i-f[m-x];
if(day1<=day2)tot=Max(day1*A,0);
else tot=Max(day2*A+(day1-day2)*B,0);
}//贪心的核心部分,具体见分析,注意要严格保证不能为负数
tot+=Max(Num*i-h[Num],0)*C;//加上学生的不愉快度
x+=sum[i];//更新
ans=Min(ans,tot);//取最小
}
cout<<ans<<endl;
return 0;
}
\(\operatorname{Update}\) \(\operatorname{On}\) \(\operatorname{2019.08.22}\)
题解 洛谷P3745 【[六省联考2017]期末考试】的更多相关文章
- 洛谷 P3745 [六省联考2017]期末考试
题目描述 有 nnn 位同学,每位同学都参加了全部的 mmm 门课程的期末考试,都在焦急的等待成绩的公布. 第 iii 位同学希望在第 tit_iti 天或之前得知所有课程的成绩.如果在第 tit_ ...
- 洛谷P3745 [六省联考2017]期末考试
传送门 题解 //Achen #include<algorithm> #include<iostream> #include<cstring> #include&l ...
- [luogu] P3745 [六省联考2017]期末考试 (贪心)
P3745 [六省联考2017]期末考试 题目描述 有 \(n\) 位同学,每位同学都参加了全部的 \(m\) 门课程的期末考试,都在焦急的等待成绩的公布. 第 \(i\) 位同学希望在第 \(t_i ...
- 【BZOJ4868】[六省联考2017]期末考试(贪心)
[BZOJ4868][六省联考2017]期末考试(贪心) 题面 BZOJ 洛谷 题解 显然最终的答案之和最后一个公布成绩的课程相关. 枚举最后一天的日期,那么维护一下前面有多少天可以向后移,后面总共需 ...
- 洛谷 P3747 [六省联考2017]相逢是问候 解题报告
P3747 [六省联考2017]相逢是问候 题目描述 \(\text {Informatik verbindet dich und mich.}\) 信息将你我连结. \(B\) 君希望以维护一个长度 ...
- [BZOJ4868][六省联考2017]期末考试(三分)
4868: [Shoi2017]期末考试 Time Limit: 20 Sec Memory Limit: 512 MBSubmit: 964 Solved: 439[Submit][Status ...
- 洛谷P3749 [六省联考2017]寿司餐厅
传送门 题解 这几道都是上周llj讲的题,题解也写得十分好了,所以直接贴了几个链接和代码. //Achen #include<algorithm> #include<iostream ...
- 洛谷 P3750 [六省联考2017]分手是祝愿
传送门 题解 //Achen #include<algorithm> #include<iostream> #include<cstring> #include&l ...
- 洛谷P3747 [六省联考2017]相逢是问候
传送门 题解 扩展欧拉定理. 线段树维护,已经全改到底了的节点就不管,不然暴力修改下去. //Achen #include<algorithm> #include<iostream& ...
随机推荐
- radio和checkbox的js勾选使用
Html: <table> <tr><th class="w1">党内职务</th><td colspan="3&q ...
- js arguments
偶然碰见一个有意思的题 <script> var length = 10; function fn() { console.log( this.length ); // 10 } var ...
- Influx Sql系列教程零:安装及influx-cli使用姿势介绍
influxdb 时序数据库,因为实际业务中使用到了,然而并没有发现有特别好的文章,完整的介绍influx sql的使用姿势,因此记录下实际开发中学习的体会,主要参考来自于官方文档 Influx Qu ...
- ubuntu samba 服务器搭建
最近总是在搭建 samba 环境,写在笔记上记录下以备后用,长时间不操作了肯定会忘记. Linux 版本:Ubuntu 18.04 具体的操作命令: 1. 安装: sudo apt-get insta ...
- appium通过index查找目标控件
2.1 通过判断控件属性获取控件 控件的所有属性都可以用作判断,比如它的text,index,resource-id是否clickable等,例如: 2.1.1 通过文本查找目标控件 1 2 el = ...
- 学习数据结构Day4
链表 之前看过了动态数组,栈和队列,虽然我们把第一个叫做动态数组,但是,他们的底层实质上还是静态数组.靠 resize来实现动态数组.而链表是真正的数据结构 链表需要一个节点. 数据存储在链表中 相当 ...
- 【Linux】进程的结构,创建,结束,以及程序转化为的进程的过程
本文内容: 1.进程的结构 2.程序转化为进程的过程 3.进程的创建 4.进程的结束 背景知识: 1.进程是计算机中处于运行的程序的实体 2.进程是线程的容器 3.程序本身只是指令,数据以及组织形式的 ...
- 记28377系列芯片中Can总线标准帧和扩展帧该怎么设置?
笔者最近在调试28377系列DSP芯片的can通讯时,遇到一个小问题,百思不得姐~ 起因是这样的,在设计一个多单元并联的系统,所有单元使用can总线进行通讯,当通讯端口,can外设,以及相关通讯协议都 ...
- Go语言 (指针)
区别于C/C++中的指针,Go语言中的指针不能进行偏移和运算,是安全指针. 要搞明白Go语言中的指针需要先知道3个概念:指针地址.指针类型和指针取值. Go语言中的指针 Go语言中的函数传参都是值拷贝 ...
- 怎么理解js的原型链继承?
前言 了解java等面向对象语言的童鞋应该知道.面向对象的三大特性就是:封装,继承,多态. 今天,我们就来聊一聊继承.但是,注意,我们现在说的是js的继承. 在js的es6语法出来之前,我们想实现js ...