【蓝桥杯每日一题】重新排序

重新排序

2024-12-8 蓝桥杯每日一题 重新排序 前缀和 差分

题目大意

给定一个数组 A 和一些查询 L i , R i Li_,R_i Li,Ri, 求数组中第 L i L_i Li至第 R i R_i Ri个元素之和。

小蓝觉得这个问题很无聊, 于是他想重新排列一下数组, 使得最终每个查 询结果的和尽可能地大。小蓝想知道相比原数组, 所有查询结果的总和最多可 以增加多少?

解题思路

首先看到题一定会想到前缀和,因为要求数组中某一个区间的和。

分析这个题,想要某些区间里的和达到最大,那么可以让那些重合计算的位置能够交换到最大的值,以此达目的。

然后就是计算每一个位置的使用使用次数,可以通过差分,这里涉及到对某些区间的一个 +1 。

最后想要给这些使用次数最多的分配到可行的最大值,那么可以通过排序,将原数组和使用的次数都进行一个排序,那么这时候就满足上述那个要求;然后分别计算两次的总和相减即可。

举例:

cpp 复制代码
1 2 3 4 5    	1 3   2 5
1 2 2 1 1 
    
排序:	
1 2 3 4 5 
1 1 1 2 2
Accepted
cpp 复制代码
//  不开 long long 见祖宗
#include <bits/stdc++.h>
using namespace std;

typedef long long ll;
const int N = 100010;
ll a[N],b[N],c[N];
ll n,m;

void diff(int l,int r) {
    b[l] += 1;
    b[r + 1] -= 1;
}

int main()
{
    cin>>n;
    for(int i = 1 ;i <= n ;i++ ) {
        cin>>a[i];
        c[i] = c[i-1] + a[i];
    }
    cin>>m;
    ll cnt = 0;
    for(int i = 1;i <= m;i++) {
        int l,r;
        cin>>l>>r;
        diff(l,r);
        cnt += c[r] - c[l-1];
    }
    for(int i = 1;i <= n;i++) {
        b[i] = b[i-1] + b[i];
    }
    sort(b+1,b+n+1);
    sort(a+1,a+n+1);

    ll pre = 0;
    for(int i = 1;i <= n;i++) {
        pre += a[i] * b[i];
    }
    cout<<pre - cnt<<endl;
    return 0;
}
思考

当时的思考已经分析到了使用次数,就差最后的一个神来之笔------排序求解。

备注

想要一起备赛的小伙伴可以私信加我的联系方式!

相关推荐
研究点啥好呢14 分钟前
Github热门项目推荐 | 创建你的像素风格!
c++·python·node.js·github·开源软件
_dindong14 分钟前
cf1091div2 C.Grid Covering(数论)
c++·算法
沫璃染墨27 分钟前
C++ string 从入门到精通:构造、迭代器、容量接口全解析
c语言·开发语言·c++
6Hzlia1 小时前
【Hot 100 刷题计划】 LeetCode 17. 电话号码的字母组合 | C++ 回溯算法经典模板
c++·算法·leetcode
计算机安禾2 小时前
【数据结构与算法】第36篇:排序大总结:稳定性、时间复杂度与适用场景
c语言·数据结构·c++·算法·链表·线性回归·visual studio
unicrom_深圳市由你创科技2 小时前
做虚拟示波器这种实时波形显示的上位机,用什么语言?
c++·python·c#
无限进步_2 小时前
【C++】电话号码的字母组合:从有限处理到通用解法
开发语言·c++·ide·windows·git·github·visual studio
C++ 老炮儿的技术栈2 小时前
GCC编译时无法向/tmp 目录写入临时汇编文件,因为设备空间不足,解决
linux·运维·开发语言·汇编·c++·git·qt
橘颂TA2 小时前
【笔试】算法的暴力美学——牛客 NC213140 :除2!
c++·算法·结构与算法
yoyobravery2 小时前
蓝桥杯第15届单片机满分
单片机·职场和发展·蓝桥杯