P1678 烦恼的高考志愿

题目背景

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

题目描述

现有 m 所学校,每所学校预计分数线是 ai​。有 n 位学生,估分分别为 bi​。

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

输入格式

第一行读入两个整数 m,n。

第二行共有 m 个数,表示 m 个学校的预计录取分数。

第三行有 n 个数,表示 n 个学生的估分成绩。

输出格式

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

输入输出样例

输入 #1复制

4 3
513 598 567 689
500 600 550

输出 #1复制

32

说明/提示

数据范围:

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

对于 100% 的数据,1≤n,m≤105,估分和录取线 ≤106 且均为非负整数。

#include <bits/stdc++.h> 
using namespace std;

long long a[100005]; // 存储输入的整数数组,最大容量为100005

int main() {
    long long m, c = 0; // m表示数组长度,c用于累计总差值
    int n;              // n表示需要查询的数字个数
    
    cin >> m; // 输入数组大小m
    cin >> n; // 输入查询次数n

    // 输入原始数组元素,从索引1开始
    for (int i = 1; i <= m; i++) {
        cin >> a[i];
    }

    // 对数组进行排序,以便后续使用二分查找
    sort(a + 1, a + m + 1);

    // 处理每个查询
    for (int i = 0; i < n; i++) {
        int x;
        cin >> x; // 输入要查询的目标值x

        int l = 1, r = m; // 初始化二分查找的左右边界

        // 二分查找最左边不小于x的位置(即第一个大于等于x的下标)
        while (l < r) {
            int mid = (l + r) >> 1; // 计算中间位置
            if (a[mid] >= x) {
                r = mid;
            } else {
                l = mid + 1;
            }
        }

        // 此时r是第一个>=x的元素下标
        // 比较a[r]与a[r-1]哪个更接近x,并累加差值
        if (abs(a[r] - x) < abs(a[r - 1] - x) || r == 1) {
            c += abs(a[r] - x);
        } else {
            c += abs(a[r - 1] - x);
        }
    }

    cout << c; // 输出所有查询结果与最近数组元素的绝对差之和
    return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值