怎么理解二分查找与二分答案(洛谷P1678烦恼的高考志愿)

拿洛谷P1678烦恼的高考志愿来理解一下二分法

那么什么情况可以用到二分法呢,在这道题目中有两个重要数据,一个是m个学校的预计录取分数线,一个是n个学生的估分成绩。

根据题意,我们需要求出与学生估计分数最相近的录取分数线,我们把录取分数线从小到大排序,这个录取分数线就是这个题目的关键拥有递增性的值。

思路,我们先找到第一个比预估成绩大的那个学校的录取分数线,那么比这个学校录取分数线低一点的那个学校,这两个学校的录取分数线和这个同学预估分数线的绝对值比较大小,小的那个就是我们所求的值。

其实我们在对录取分数线排序之后就可以开始二分查找了,我们看到输入输出的样例,把录取分数线从小到大排序(1)513,(2)567,(3)598,(4)689,第一位同学的预估成绩是500,一共有四个学校的录取分数,我们要取中间的值和这个同学的成绩比较【(1)+(4)】/2=(2)(计算机的算法,整数计算,计算结果没用小数),500和(2)567比较,567更大,所以继续对半取进行比较【(1)+(2)】/2=(1),(1)513和500比,(1)更大,但是已经无法继续二分了(判断条件),因为(1)已经到最小了,所以和500最接近的值就是513(后面按照题目意思取一下绝对值)。

我们再看一下第二位同学,他的预估超级是600,【(1)+(4)】/2=(2),600和(2)567比较,600更大,(因为600比(1)和(2)都大,所以继续二分就不用包含(2)了),所以二分【(3)+(4)】/2=(3),(3)598和600比,600更大,继续二分,这里600已经比598大了,所以(3)就不在二分的范围里,继续二分【(4)+(4)】/2=【4】,【4】689比600大,找到第一个比600大的数。

第三位同学,他的预估超级是550,【(1)+(4)】/2=(2),550和(2)567比较,(2)567更大,所以二分【(1)+(2)】/2=(1),(1)513和550,550更大,所以(2)就是第一个比550大的数。

注意判断结束的条件和数组里的小细节!!!

接下来附上代码

#include<bits/stdc++.h>
using namespace std;
#define int long long
int m,n;
int a[100001],b[100001];//a是学校的录取分数线,b是学生的预估成绩
signed main()
{
    cin>>m>>n;
    for(int i=1;i<=m;i++)
        cin>>a[i];
    for(int i=1;i<=n;i++)
        cin>>b[i];
    sort(a+1,a+1+m);
    a[0]=a[1];//因为a[0]最开始等于0,要考虑到a[1]>b[i]
    int ans=0;
    for(int i=1;i<=n;i++)
    {
        int l=1,r=m;
        while(l<r)//当L<r,继续找,直到l==r,则找到了那个数字
        {
            int mid=(l+r)>>1;//相当于除2
            if(a[mid]<=b[i])l=mid+1;
            else r=mid;
        }
        ans+=min(abs(a[l]-b[i]),abs(a[l-1]-b[i]));
    }
    cout<<ans;
    return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值