当前位置: 首页 > article >正文

「数组」离散化 / Luogu B3694(C++)

目录

概述

思路

算法过程

复杂度

Code


概述

Luogu B3694:

给定一个长度为 n 的数列 aa。定义 rank(i) 表示数列 a 中比 ai 小的不同数字个数再加一。

对 1≤i≤n,现在请你求出所有的 rank(i)。

输出格式

对每组数据,输出一行 n 个整数,用空格隔开,依次示 rank(1)到 rank(n)。

我们来讲一个十分常用的小算法:离散化。 

离散化并不单独作为一个算法出现,而是常作为一个预处理步骤。

所谓离散化,就是使用数组元素的相对排名替代原始数组元素。

例如,

     i  0 1 2 3 4
nums[i] 9 7 4 0 1000
         ↓
nums[i] 4 3 2 1 5

这样我们在保证元素的相对关系的前提下,使得元素的分散程度减小了。

换句话说,如果元素分布范围很大,或者元素是浮点数以及其他类型,在不关注元素的绝对性质时,用离散化处理会使得我们关注的范围更加紧凑。 

离散化在许多只关注元素相对性质的算法里有广泛的应用。

思路

一共有三个数组:原始数组,排序后的原数组,离散化数组。

先复制一份原始数组,对其进行排序,这样(元素下标-数组头指针)就得到了元素在原数组的排名。 

原始数组遍历,对于第i个元素,它在排序后数组中处于pos位置,则离散化数组[i]=pos。

需要注意的是:在考虑重复元素的只占排名时,我们要对排序后数组进行去重。

算法过程

使用C++STL算法库:

sort()排序数组。

unqiue()对排序后数组去重。

lower_bound()利用二分查找找到排序后数组中元素的位置。

排序数组b,

sort(b,b+n);

排序后对数组去重,得到排序后去重数组b[i],

int* end=unique(b,b+n); 

unqiue会将数组中的重复元素转移至数组末位,返回去重后数组无重复区的后一个位置。 

二分查找得到a[i]在b中的位置,减去b得到相对排名,

for(int i=0;i<n;i++)
        cout<<lower_bound(b,end,a[i])-b+1<<' ';

对于本题,直接输出离散结果即可,另外,题目要求的排序从1开始。 

对于二分查找lower_bound的内部实现:「数组」二分查找模版|二段性分析|恢复二段性 / LeetCode 35|33|81(C++)

对于数组去重unqiue的内部实现:

「数组」数组双指针算法合集:二路合并|逆向合并|快慢去重|对撞指针 / LeetCode 88|26|11(C++)

复杂度

时间复杂度:O(nlogn)

空间复杂度:O(n)

Code

#include <bits/stdc++.h>
using namespace std;
void solve(){
    int n;cin>>n;
    int a[n],b[n];
    for(int i=0;i<n;i++){
        cin>>a[i];
        b[i]=a[i];
    }
    sort(b,b+n);
    int* end=unique(b,b+n);
    for(int i=0;i<n;i++)
        cout<<lower_bound(b,end,a[i])-b+1<<' ';
    cout<<endl;
}
int main(){
    int t;cin>>t;
    while(t--)solve();
}

http://www.kler.cn/news/323022.html

相关文章:

  • 畅阅读微信小程序
  • RHCS认证-Linux(RHel9)-Ansible
  • 代码随想录Day 58|拓扑排序、dijkstra算法精讲,题目:软件构建、参加科学大会
  • Linux操作系统中MongoDB
  • 前端开发之代理模式
  • Cilium + ebpf 系列文章- (六)Cilium-BGP与分发-EXTERNAL-IP
  • vue3中< keep-alive >页面实现缓存及遇到的问题
  • 【深度学习】深度学习框架有哪些及其优劣势介绍
  • 【CSS】透明度 、过渡 、动画 、渐变
  • JAVAEE如何实现网页(jsp)间的数据传输?一文总结
  • 2024 icpc 第二场网络赛题解
  • vue-cli,element-plus,axios,proxy
  • 31 变量的访问方式(直接和间接),内存地址(32 位和 64 位),指针的概念与定义,取址与取值运算符( 与 *)
  • Spark Streaming 容错机制详解
  • 【Docker】如何让docker容器正常使用nvidia显卡
  • 处理execl表格的库----openpyxl
  • C++ 文件I/O流
  • 【SpringBoot详细教程】-03-整合Junit【持续更新】
  • 代码随想录Day 57|prim算法和kruskal算法精讲,题目:寻宝
  • 提升效率的秘密武器选择指南
  • PTH原理 补丁+工具
  • Java项目——苍穹外卖总结
  • Linux usb hub阅读
  • 【学习】电脑上有多个GPU,命令行指定GPU进行训练。
  • C语言习题~day33
  • 【Unity保龄球项目】的实现逻辑以及代码解释
  • Python Daphne库:ASGI服务的高效Web服务器
  • 使用FFmpeg压缩MP3格式音频
  • 利用模糊综合评价法进行数值评分计算——代码实现
  • 基于Java开发的(控制台)模拟的多用户多级目录的文件系统