LOJ2481 Sure Bet

题意

有两个组A,B,两组各有n个元素。
两组中的元素可以任选,每选取一个数,无论是从A组还是B组中选出来的,花费1的代价。
每个元素最多只能被选1次。
获利是A组的选出的元素之和 和 B组的选出的元素之和 的最小值。
问最大的 获利减去代价 是多少。
可以一个数都不选。

解题思路

肯定是从大到小排序后选A组的头几个,和B组的头几个最优。
f [ i ] f[i] f[i]表示A组选了i个,的最大的 获利减去代价。
f [ i ] = m a x { m i n ( s a [ i ] , s b [ j ] ) − i − j } f[i]=max\{min(sa[i],sb[j])-i-j\} f[i]=max{min(sa[i],sb[j])ij}
其中, s a [ i ] sa[i] sa[i]表示A组前i项的和, s b [ i ] sb[i] sb[i]类似。
三分是可以做的。因为当j增大的时候, s b [ j ] sb[j] sb[j]要么 ≥ s a [ i ] ≥sa[i] sa[i],要么 &lt; a [ i ] &lt;a[i] <a[i]
但是 f [ i ] f[i] f[i]是有决策单调性的。
可以证明。反证法,分6种情况就好。
时间复杂度 O ( n ) O(n) O(n)

证明决策单调性

情况① s b j ≤ s b j ′ ≤ s a i &lt; s a i + 1 sb_j≤sb_{j&#x27;}≤sa_i&lt;sa_{i+1} sbjsbjsai<sai+1 矛盾。
情况② s b j ≤ s a i ≤ s b j ′ &lt; s a i + 1 sb_j≤sa_i≤sb_{j&#x27;}&lt;sa_{i+1} sbjsaisbj<sai+1 矛盾。
情况③ s b j ≤ s a i &lt; s a i + 1 ≤ s b j ′ sb_j≤sa_i&lt;sa_{i+1}≤sb_{j&#x27;} sbjsai<sai+1sbj 矛盾。
情况④ s a i ≤ s b j ≤ s b j ′ &lt; s a i + 1 sa_i≤sb_j≤sb_{j&#x27;}&lt;sa_{i+1} saisbjsbj<sai+1 矛盾。
情况⑤ s a i ≤ s b j &lt; s a i + 1 ≤ s b j ′ sa_i≤sb_j&lt;sa_{i+1}≤sb_{j&#x27;} saisbj<sai+1sbj 矛盾。
情况⑥ s a i &lt; s a i + 1 ≤ s b j ≤ s b j ′ sa_i&lt;sa_{i+1}≤sb_j≤sb_{j&#x27;} sai<sai+1sbjsbj 矛盾。
用反证法证明的时候,如果不等式等式两边相等,那么当作矛盾处理。

代码

#include<iostream>
#include<cstdio>
#include<cmath>
#include<algorithm>
#include<cstring>
#define N 100010
#define DB double
#define fo(i,a,b) for(i=a;i<=b;i++)
using namespace std;
int i,j,n,s[N];
DB f[N],a[N],b[N],sa[N],sb[N];
DB ans;
bool cmp(DB a,DB b){return a>b;}
DB qz(int i,int j){return min(sa[i],sb[j])-i-j;}
int main(){
	freopen("2481.in","r",stdin);
	scanf("%d",&n);
	fo(i,1,n)scanf("%lf%lf",&a[i],&b[i]);
	sort(a+1,a+n+1,cmp);
	sort(b+1,b+n+1,cmp);
	fo(i,1,n){
		sa[i]=sa[i-1]+a[i];
		sb[i]=sb[i-1]+b[i]; 
	}
	fo(i,1,n)f[i]=-1e15;ans=0;
	j=1;
	fo(i,1,n){
		while(j<n&&qz(i,j)<qz(i,j+1))j++;
		ans=max(ans,qz(i,j));
	}
	/*fo(j,1,n){
		fo(i,1,n)
		if(qz(i,j)>f[i]){
			f[i]=qz(i,j);
			s[i]=j;
		}
		if(j>n-1){
			fo(i,1,n)printf("%d ",s[i]);printf("\n");
	    }	
	}
	fo(i,1,n)if(s[i]==s[i-1])printf("%d\n",i);
	fo(i,1,n)ans=max(ans,f[i]);*/
	printf("%.4lf",ans);
	return 0;
}
1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md或论文文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。 5、资源来自互联网采集,如有侵权,私聊博主删除。 6、可私信博主看论文后选择购买源代码。 1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md或论文文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。 5、资源来自互联网采集,如有侵权,私聊博主删除。 6、可私信博主看论文后选择购买源代码。 1、资源项目源码均已通过严格测试验证,保证能够正常运行; 2、项目问题、技术讨论,可以给博主私信或留言,博主看到后会第一时间与您进行沟通; 3、本项目比较适合计算机领域相关的毕业设计课题、课程作业等使用,尤其对于人工智能、计算机科学与技术等相关专业,更为适合; 4、下载使用后,可先查看README.md或论文文件(如有),本项目仅用作交流学习参考,请切勿用于商业用途。 5、资源来自互联网采集,如有侵权,私聊博主删除。 6、可私信博主看论文后选择购买源代码。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值