[CSP-S模拟测试]:爬(贪心)

题目传送门(内部题134)


输入格式

  第一行两个数$N,L$。
  接下来$N$行每行两个数$A_i,B_i$。
  接下来$N$行每行一个整数$C_i$。


输出格式

  一行一个整数表示答案,无解输出$-1$。


样例

样例输入1:

3 9
6 3
5 2
3 1
2
2
2

样例输出1:

2

样例输入2:

5 20
3 2
4 2
6 3
8 4
10 5
4
2
3
4
5

样例输出2:

-1


数据范围与提示

  对于$40\%$的数据,$N\leqslant 1,000$;
  对于额外$20\%$的数据,$B_i=0$;
  对于额外$20\%$的数据,$C_i=0$;
  对于$100\%$的数据,$n\leqslant 10^5,0\leqslant A_i,B_i,C_i\leqslant 10^9$。


题解

瞎打都能过的题……

考场打了个假贪心$A$掉了,然而打部分分的$KX$表示不服(因为数据中没有给部分分),于是他就跟教练要求要出数据,于是……

上面的$KX$有他博客的链接,欢迎大家消费$+$吐槽。

考场上的贪心实际上已经很接近正解了。

首先,按$A_i-B_i$排序,在爬不上去的情况下一定是优先选择$A_i-B_i$最大的。

然后发现其实有的时候可以用一个$A_i$很大的“冲”上去,所以开一个大根堆就好了。

但是还有一个情况,我们可以用一些$A_i-B_i$比较小的“积蓄能量”,然后用一个$A_i-B_i$较大但是$A_i$更大的“冲”上去。

然后发现只与最后一个有关,所以可以只考虑最后一个选什么,用倍增思想维护即可。

时间复杂度:$\Theta(n\log n)$。

期望得分:$100$分。

实际得分:$100$分。


代码时刻

#include<bits/stdc++.h>
using namespace std;
struct rec{int A,B,delta;}e[100001];
int N;
long long L;
long long C[100001],S[100001],F[20][100001];
int lg[100001];
int limits;
bool vis[100001];
int ans=0x3f3f3f3f;
bool cmp(rec a,rec b){return a.delta>b.delta;}
int cal(int l,int r){return min(F[lg[r-l+1]][l],F[lg[r-l+1]][r-(1<<lg[r-l+1])+1]);}
int main()
{
	scanf("%d%lld",&N,&L);vis[0]=1;
	for(int i=2;i<=N;i++)lg[i]=lg[i>>1]+1;
	for(int i=1;i<=N;i++)
	{
		scanf("%d%d",&e[i].A,&e[i].B);
		e[i].delta=e[i].A-e[i].B;
	}
	sort(e+1,e+N+1,cmp);limits=N;
	for(int i=1;i<=N;i++)if(e[i].delta<0){limits=i-1;break;}
	for(int i=1;i<=N;i++)
	{
		scanf("%d",&C[i]);
		C[i]+=C[i-1];
		S[i]=S[i-1]+e[i].delta;
		F[0][i]=S[i]+e[i].delta-C[i];
		if(vis[i-1]&&C[i]<S[i])vis[i]=1;
	}
	for(int i=1;i<=lg[N];i++)
		for(int j=1,s=(1<<(i-1));j<=N-(1<<j)+1;j++)
			F[i][j]=min(F[i-1][j],F[i-1][j+s]);
	long long now;
	int pos;
	for(int i=1;i<N;i++)
	{
		now=L+e[i].delta-e[i].A;
		pos=lower_bound(S+1,S+limits+1,now)-S;
		if(pos==limits+1)continue;
		if(vis[i-1]&&pos>i)
			if(cal(i,pos)>e[i].delta)
				ans=min(ans,pos);
	}
	for(int i=N,mx=0;i>=0;i--)
	{
		if(vis[i]&&S[i]+mx>=L)
			ans=min(ans,i+1);
		mx=max(mx,e[i].A);
	}
	if(ans==0x3f3f3f3f)puts("-1");
	else printf("%d\n",ans);
	return 0;
}


rp++

原文地址:https://www.cnblogs.com/wzc521/p/11824354.html

时间: 2024-08-30 10:42:56

[CSP-S模拟测试]:爬(贪心)的相关文章

2018冬令营模拟测试赛(三)

2018冬令营模拟测试赛(三) [Problem A]摧毁图状树 试题描述 输入 见"试题描述" 输出 见"试题描述" 输入示例 见"试题描述" 输出示例 见"试题描述" 数据规模及约定 见"试题描述" 题解 这题没想到贪心 QwQ,那就没戏了-- 贪心就是每次选择一个最深的且没有被覆盖的点向上覆盖 \(k\) 层,因为这个"最深的没有被覆盖的点"不可能再有其它点引出的链覆盖它了,而它又

2018冬令营模拟测试赛(十九)

2018冬令营模拟测试赛(十九) [Problem A]小Y 试题描述 输入 见"试题描述" 输出 见"试题描述" 输入示例 见"试题描述" 输出示例 见"试题描述" 数据规模及约定 见"试题描述" 题解 目前未知. 这题目前就能做到 \(O(n \sqrt{M} \log n)\),其中 \(M\) 是逆序对数,然而会被卡 \(T\):当然这题暴力可以拿到和左边那个算法一样的分数,只要暴力加一个剪枝:当左

[考试反思]0929csp-s模拟测试55:沦陷

菜得过分. 面对T1的大板子不知所措,然后T2的贪心不小心把排序语句删了... T1这种大模板啊...其实我是觉得我能打出来的,然后先用一个小时码了一个2k. 然后做T2想贪心就出来了.十分钟码完T3暴力之后回T1打对拍瞬间爆炸. 于是又重新打了一个2k,WA0.对拍发现. 然后考试就没几分钟了交暴力走了. 不要打完就跑,记得早点对拍改进思路. T1: 的确是挺裸的线段树.离散化或者权值线段树都可以. 但是考场上两个都打出来都死了. 最后用离散化A的. 1 #include<cstdio> 2

[考试反思]1002csp-s模拟测试56:凌乱

放假回来状态回升??(玩够了-但是稍困) T1打的不完全对,但是过掉了.很快的想到了二分吧喇叭啦.. 然后T2也挺快想出来了但是挂细节没发现,考试快结束的时候才发现出锅了. 改了过来是正解,但是出题人无良卡了线段树强制树状数组,T了一个子任务,卡常到飞起. T3暴力没什么问题. 卡常是一种习惯.要注意题目数据范围观察是否卡常. T1: 所有的决策都是一条一次函数. 分两类,斜率正或斜率非负. 如果第二类的直线里有在T=0时符合要求的,那么答案就是0,所以check(0)一下. 如果非负的直线都在

模拟测试68,69

68: 32 AlpaCa 41 03:08:20 31 02:46:16 30 02:46:28 102 03:08:20 69: 28 AlpaCa 20 02:51:15 60 03:05:32 0 01:39:45 80 03:05:32 彻底挂掉了呢. 不过也还好吧,至少之后的考试不会有那么大压力了吧(出第一机房是肯定的事了),那利用之后几场考试就调整好状态,下次再来嘛. 但是最近炸了那么多场还是要反思一下,其实不只是心态的问题,自身实力,考试技巧,时间分配等方面好像都有点问题. 69

csp-s模拟测试92

csp-s模拟测试92 关于$T1$:最短路这一定建边最短路. 关于$T2$:傻逼$Dp$这一定线段树优化$Dp$. 关于$T3$:最小生成树+树P+换跟一定是这样. 深入(?)思考$T1$:我是傻逼. 深入(?)思考$T2$:我是天才! 深入(?)思考$T3$:我是天才! 100 01:14:19 10 02:36:26 15 03:19:48 125 03:19:48 事实证明我是傻逼. A. 数列 故意删掉负数读入$Rp--$ $Exgcd$,对绝对值大力分类讨论,一次函数求最值. 考场死

模拟测试(vj)

做这份模拟测试,已经崩溃了,英文看不懂,题意理解错.到结束了只a了第一题,人生陷入了低谷,于是花了一天的时间终于把不会的弄明白了,在这里写一份总结~ T1,简单的模拟,如果打枪打中一支鸟,将这个位置设为0,并向两边扩散,注意这个位置一定要有鸟. 代码~ #include<bits/stdc++.h> using namespace std; int a[30000]; int n,m; int main() { cin>>n; for(int i=1;i<=n;i++) ci

Android单元测试与模拟测试详解

测试与基本规范 为什么需要测试? 为了稳定性,能够明确的了解是否正确的完成开发. 更加易于维护,能够在修改代码后保证功能不被破坏. 集成一些工具,规范开发规范,使得代码更加稳定( 如通过 phabricator differential 发diff时提交需要执行的单元测试,在开发流程上就可以保证远端代码的稳定性). 2. 测什么? 一般单元测试: 列出想要测试覆盖的异常情况,进行验证. 性能测试. 模拟测试: 根据需求,测试用户真正在使用过程中,界面的反馈与显示以及一些依赖系统架构的组件的应用测

使用Post方法模拟登陆爬取网页(转)

使用Post方法模拟登陆爬取网页 最近弄爬虫,遇到的一个问题就是如何使用post方法模拟登陆爬取网页.下面是极简版的代码: import java.io.BufferedReader; import java.io.InputStreamReader; import java.io.OutputStreamWriter; import java.io.PrintWriter; import java.net.HttpURLConnection; import java.net.URL; impo