P1678 烦恼的高考志愿---洛谷(二分)

题目链接

https://www.luogu.com.cn/problem/P1678

题目描述

题目背景

计算机竞赛小组的神牛 V 神终于结束了高考,然而作为班长的他还不能闲下来,班主任老 t 给了他一个艰巨的任务:帮同学找出最合理的大学填报方案。可是 v 神太忙了,身后还有一群小姑娘等着和他约会,于是他想到了同为计算机竞赛小组的你,请你帮他完成这个艰巨的任务。

题目描述

现有m(m≤100000) 所学校,每所学校预计分数线是 ai​(ai​≤10^6)。有 n(n≤100000) 位学生,估分分别为 bi​(bi​≤10^6)。

根据 n 位学生的估分情况,分别给每位学生推荐一所学校,要求学校的预计分数线和学生的估分相差最小(可高可低,毕竟是估分嘛),这个最小值为不满意度。求所有学生不满意度和的最小值。

输入格式

第一行读入两个整数 m,n。m 表示学校数,n 表示学生数。

第二行共有 m 个数,表示 m 个学校的预计录取分数。第三行有 n 个数,表示 n 个学生的估分成绩。

输出格式

输出一行,为最小的不满度之和。

输入输出样例

输入

4 3
513 598 567 689
500 600 550

 输出

32

说明/提示 

数据范围:

对于 30% 的数据,1≤n,m≤1000,估分和录取线 ≤10000;

对于 100% 的数据,1≤n,m≤100000,估分和录取线 ≤1000000。

思路分析 

这道题题目大意就是对于每一位学生都去与每一所学校作比较,最后找出每一位学生的分数与各自学校分数线差最小的分差和。如果每个学生都与每个学校分数线比较,那么二重循环后要搜索的最大数据量为m*n,在本题中则是最多可能要判断10000000000次,时间复杂度较高,那么这种涉及比较且数据量又大的,我们可以采用二分法。

二分法简介

C++语言中的二分查找法

算法:当数据量很大适宜采用该方法。采用二分法查找时,数据需是排好序的。

基本思想:假设数据是按升序排序的,对于给定值x,从序列的中间位置开始比较,如果当前位置值等于x,则查找成功;若x小于当前位置值,则在数列的前半段中查找;若x大于当前位置值则在数列的后半段中继续查找,直到找到为止。

假如有一组数为3,12,24,36,55,68,75,88要查给定的值24.可设三个变量front,mid,end分别指向数据的上界,中间和下界,mid=(front+end)/2.

1.开始令front=0(指向3),end=7(指向88),则mid=3(指向36)。因为mid>x,故应在前半段中查找。

2.令新的end=mid-1=2,而front=0不变,则新的mid=1。此时x>mid,故确定应在后半段中查找。

3.令新的front=mid+1=2,而end=2不变,则新的mid=2,此时a[mid]=x,查找成功。

如果要查找的数不是数列中的数,例如x=25,当第三次判断时,x>a[mid],按以上规律,令front=mid+1,即front=3,出现front>end的情况,表示查找不成功。

这道题与之不同的地方是我们不是要在一组数据中查找一个数,而是找到与之差值最小的数,那么当我们将m个学校分数线从小到大排好序后,便可以采用二分法,那么我们想要找到与某位学生分数X相差最小的分数线S,只要在已经排好序的学校分数线中找到S[i]<=X<=S[i+1],那么最小分差值便是min(abs(X-S[i]),abs(X-S[i+1]))(abs是指取绝对值)。

另外还需注意的是有可能学生的分数在所有学校分数线中是<=最低分或者>=最高分这种情况,所以在最开头要加上对这种情况的特殊判断!

代码贴这

#include<iostream>
#include<algorithm> 
#include<cmath>
using namespace std;
int main()
{
	int m,n;
	cin>>m>>n;
	int*s=new int[m];
	int*st=new int[n];
	int sum=0;//统计所有学生与相应学校的最小分差和
	for(int i=0;i<m;i++)
	{
		cin>>s[i];
	}
    //排序
	sort(s,s+m);
	for(int i=0;i<n;i++)
	{
		cin>>st[i];
	}
	for(int i=0;i<n;i++)
	{
        //判断边缘的特殊情况
		if(st[i]<=s[0])
		{
			sum+=abs(st[i]-s[0]);
			continue;
		}
		else if(st[i]>=s[m-1])
		{
			sum+=abs(st[i]-s[m-1]);
			continue;
		}
        //如果不是边缘特殊情况,再开始二分查找
		int l=0,r=m-1;
		while(1)
		{
			int md=(l+r)/2;
			if(st[i]<s[md])
			{
				r=md;
			}
			else if(st[i]>s[md+1])
			{
				l=md;
			}
			else
			{
				sum+=min(abs(st[i]-s[md]),abs(st[i]-s[md+1]));
				break;
			}
		}
	}
	cout<<sum<<endl;
	return 0;
}

有什么需要改进的请大家不吝赐教咯

  • 4
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值