洛谷P1631 序列合并 二分做法

洛谷P1631

之前也写的二分一直过不了,以为不行,然后看了这篇laorui的题解后发现只要把数组开大一点就能过。

做了一点优化之后决定写一篇题解 因为他的代码过于清奇

思路就是二分去找数,只要这里面两两求和的对满足有n个就是true,否则就是false,因为单调增的原因只要找n个数所以judge只需要n的复杂度最后复杂度就是nlogm

但是要考虑的是可能二分judge里的退出条件设置小了,可能提前退出了,前面选中的对可能有重复,小的对可能在后面就选不到小的了。

所以得扩大退出条件,大胆假设只需要二倍的n就行了,虽然不知道怎么证明。hoho不过总之是过了。

以及要注意二分之后的目标数组不一定是答案,因为最后在judge里跑的不一定是正确的数。

还有就是二分的初始值,左端点是等于a[0]+b[0],右端点只需要等于a[d]+b[d]就行了,d=ceil(sqrt(n))-1,比这个数大的时候是显然会有超过n对的。

主要看代码

#include<bits/stdc++.h>
using namespace std;
const int max_n=2e5+5;
int a[max_n],b[max_n];
int ans[max_n];
int n,cnt,lim;
bool judge(int max1)
{
	int i,j;
	cnt=0;
	for(i=0;i<n;i++){
		for(j=0;j<n;j++){
			if(a[i]+b[j]<=max1){
				ans[cnt++]=a[i]+b[j];
				if(cnt>=lim) return true;
			}else{
				break;//利用单调性保证复杂度
			}
		}
	}
	if(cnt>=n) return true;
	else return false;
}
int main()
{
	int i,j;
	scanf("%d",&n);
	lim=n<<1;//lim退出条件选两倍大
	for(i=0;i<n;i++){
		scanf("%d",a+i);
	}
	for(j=0;j<n;j++){
		scanf("%d",b+j);
	}
	int d=ceil(sqrt(n))-1;//求右端点的位置
	int l=a[0]+b[0],r=a[d]+b[d],mid;
	int ans1=0;
	while(l<=r){
		mid=l+((r-l)>>1);
		if(judge(mid)){
			ans1=mid;
			r=mid-1;
		}else{
			l=mid+1;
		}
	}
    
	//结束之后要拿ans1再跑一遍,因为最后执行的mid不一定是满足的
	judge(ans1);
	
	sort(ans,ans+cnt);//排序之后输出前n个
	for(i=0;i<n;i++){
		printf("%d ",ans[i]);
	}
	return 0;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值