一道有关逆序对的算法题---归并、树状数组、线段树三种解法

2023-10-31

这阵子巨忙,呜呜呜....

上周末打了第十四届蓝桥杯的校内赛,总的来说,题目是很简单的,不过其中的压轴题挺有意思,所以我打算来一发博客记录记录这道算法题。废话不多说了,上题:

问题描述

        小蓝有一个序列 a[1], a[2], ..., a[n],每次可以交换相邻的两个元素,代价为两个元素中较大的那个。
  请问,要通过交换将序列变为从小到大递增的序列,总代价最少为多少?

输入格式 

        输入一行包含一个整数 n ,表示序列长度。
        第二行包含 n 个整数,表示给定的序列。

输出格式

        输出一行包含一个整数,表示最少代价的值。

样例输入1

4
1 5 2 1
样例输出1

12
评测用例规模与约定

        对于 30% 的评测用例,1 <= n <= 1000, 1 <= a[i] <= 1000。
        对于 60% 的评测用例,1 <= n <= 50000, 1 <= a[i] <= 50000。
        对于所有评测用例,1 <= n <= 1000000, 1 <= a[i] <= 1000000。

思路:

首先题目说的是,每次交换相邻的两个,取两者较大的作为代价,同时要保证最后的结果是最小的。 举个例子: 1 5 2 1,我们要变成 1 1 2 5 这样才符合题目要求对吧 ? 那么怎么让他代价最小呢? 拿 1 5 2 1来看,我们设ans = 0 ,一直看到 5 比 2 大,要交换一下 ,ans+=5,同时变成 1 2 5 1 ,5又比1 大,ans+=5, 变成1 2 1 5 ,这样子我们成功解决掉最大的数5了,再也不用加到它的值了,然后看2 与 1 交换 ans+=2 变成 1 1 2 5 ,就是我们想要的答案了。所以其实我们会发现,5后面比他小的数有2个 ans+=(5*2);  2后面有 1个比他小 ans+= (2*1); 不就是把问题转换为 求每个数后面有多少个比它大的(其实就是逆序对),然后将逆序对的个数乘以这个数的值,就是答案了!!!

所以我们可以先来一波暴力解法:

public static void main(String[] args) {
		// TODO 自动生成的方法存根
		Scanner cin = new Scanner(new BufferedReader(new InputStreamReader(System.in)));
		int n = cin.nextInt();
		int num[] = new int[n];
		for(int i = 0;i<n;i++) {
			num[i] = cin.nextInt();
		}
		long ans = 0;
		for(int i = 0;i<n;i++) {
			for(int j = i+1;j<n;j++) {
				if(num[i] > num[j]) {
					ans+=num[i];
				}
			}
		}
		System.out.println(ans);
	}

当然,这个解法O(n²)必定会超时,不过可以骗一些分,所以考虑使用其他解法。

一、归并算法找出逆序对

大家先想一想,上面的暴力解法,最大的问题就是把比num[i]大的个数全都遍历了,导致的复杂度增加,所以我们可以利用归并排序算法,在合并过程中,有一边肯定比另一边的大的特点来简化时间复杂度。具体做法看图的例子:

我们拿一个数组num[1,5,40,7,19,10]来从大到小排序看:我们肉眼就知道逆序对是4 (<40,7><40,19><40,10><19,10>)

 具体看代码:

public static void finish(int l,int r,int mid) {
		int i = l;
		int j = mid;
		int p = l;
		while(i < mid && j <= r) {
			if(num[i] <= num[j]) {
				temp[p++] = num[j++];
			}
			else { //如果右边有比它大的,因为右边已经排序了,所以直接乘以个数。
				ans += (num[i]* (r - j + 1));
				temp[p++] = num[i++];
			}
		}
		while( i< mid) {
			temp[p++] = num[i++];
		}
		while(j <= r) {
			temp[p++] = num[j++];
		}
		for(int start = l; start <= r;start++) {
			num[start] = temp[start];
		}
	}
	public static void fenzhi(int l,int r) {
		if(l < r) {
			int mid = (l+r)/2;
			fenzhi(l,mid);
			fenzhi(mid+1 , r);
			finish(l,r,mid+1);
		}
	}
	public static int n,temp[],num[];
	public static long ans = 0;
	public static void main(String[] args) {
		// TODO 自动生成的方法存根
		Scanner cin = new Scanner(new BufferedReader(new InputStreamReader(System.in)));
		n = cin.nextInt();
		num = new int[n];
		for(int i = 0;i<n;i++) {
			num[i] = cin.nextInt();
		}
		temp = new int[n];
		fenzhi(0,n-1);
		System.out.println(ans);
	}

二、利用树状数组和线段树来解决

其实这个也挺好理解,我们先要进行离散化,为啥离散化呢?原因就是:比如 {1,3,9,100,5}这5个数,我们如果我们创一个数组来保存他们的值,那么需要开到101欸,只要5个数开到101很亏,所以我们要用离散化来解决这个问题,我们给它们一个下标pi,value{1,3,9,100,5} ->pi {1,2,3,4,5} 然后按照value值来排序,得到的就是 value{1,3,5,9,100} -> pi{1,2,5,3,4},我们每次遍历数组把pi放进去,然后查一下1~pi-1放进了多少个,就可以知道有多少个逆序对了,再乘以数组就ok.

public static class stu{
		int pi;
		int value; //离散化
		public stu(int pi, int value) {
			this.pi = pi;
			this.value = value;
		}
	}
	public static int lowbit(int x) {
		return x & (-x);
	}
	public static void add(int x) {
		while(x <= n) {
			BT[x] ++;
			x += lowbit(x);
		}
	}
	public static int get(int x) {
		int ans = 0;
		while(x > 0) {
			ans += BT[x];
			x -= lowbit(x);
		}
		return ans;
	}
	public static int num[];
	public static int BT[],n;
	public static stu tree[];
	public static long ans = 0;
	public static void main(String[] args) {
		// TODO 自动生成的方法存根
		Scanner cin = new Scanner(new BufferedReader(new InputStreamReader(System.in)));
		n = cin.nextInt();
		num = new int[n];
		for(int i = 0;i<n;i++) {
			num[i] = cin.nextInt();
		}
		tree = new stu[n];
		BT = new int[n+1];
		for(int i = 0;i<n;i++) {
			tree[i] = new stu(i+1, num[i]);
		}
		Arrays.sort(tree,new Comparator<stu>() {

			@Override
			public int compare(stu o1, stu o2) {
				// TODO 自动生成的方法存根
				return o1.value == o2.value ? o1.pi - o2.pi : o1.value - o2.value;
			}
		});
		//序列化
		int ans = 0;
		for(int i = 0;i<n;i++) {
			add(tree[i].pi);
			ans += (tree[i].value*(i - get(tree[i].pi - 1))); //查一下有多少个比他小的
		}
		System.out.println(ans);
	}

线段树也是一样的思路,就是变成了线段树的写法

public static PrintWriter pw  =new PrintWriter(new BufferedWriter(new OutputStreamWriter(System.out)));
	public static StreamTokenizer st = new StreamTokenizer(new BufferedReader(new InputStreamReader(System.in)));
	public static int Int(){
		try {
			st.nextToken();
		} catch (IOException e) {
			// TODO 自动生成的 catch 块
			e.printStackTrace();
		}
		return (int)st.nval;
	}
	public static class stu{
		int pi;
		int value;
		public stu(int pi, int value) {
			super();
			this.pi = pi;
			this.value = value;
		}
	}
	public static class MyComparator<T> implements Comparator<T>{
		@Override
		public int compare(T o1, T o2) {
			// TODO 自动生成的方法存根
			stu a1 = (stu)o1;
			stu a2 = (stu)o2;
			if(a1.value == a2.value) {
				return a1.pi - a2.pi;
			}
			else {
				return a1.value - a2.value;
			}
		}
	}
	public static void build(int l,int r,int pi) {
		Arrays.fill(tree, 0);
	}
	public static int search(int l,int r,int pi,int nx ,int ny ) {
		if(nx <= l && r <= ny) {
			return tree[pi];
		}
		int mid = (l+r)/2;
		int ans = 0;
		if(mid >= nx) {
			ans += search(l,mid,pi*2,nx,ny);
		}
		if(mid < ny) {
			ans += search(mid+1,r,pi*2+1,nx,ny);	
		}
		return ans;
	}
	public static void add(int l,int r,int pi,int fill) {
		if(l > r) {
			return ;
		}
		if(l == fill && r == fill) {
			tree[pi] ++;
			return ;
		}
		int mid = (l + r)/2;
		if(fill <= mid) {
			add(l,mid,pi*2,fill);
		}
		else {
			add(mid+1,r,pi*2+1,fill);
		}
		tree[pi] = tree[pi*2] + tree[pi*2+1];
	}
	public static int n,tree[],lazy[];
	public static stu mp[];
	public static void main(String[] args) {
		// TODO 自动生成的方法存根
		n = Int();
		mp = new stu[n];
		for(int i = 0;i<n;i++) {
			int num = Int();
			mp[i] = new stu(i+1,num);
		}
		Arrays.sort(mp,new MyComparator<stu>());
		pw.flush();
		tree = new int[n*4+1];
		lazy = new int[n*4+1];
		build(1,n,1);
		long ans = 0;
		for(int i = 0;i< n;i++) {
			add(1,n,1,mp[i].pi);
			int kk = 0;
			if(mp[i].pi-1 > 0) {
				kk = search(1,n,1,1,mp[i].pi-1);		
			}
			ans += (mp[i].value * (i - kk) );
		}
		pw.println(ans);
		pw.flush();
	}

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

一道有关逆序对的算法题---归并、树状数组、线段树三种解法 的相关文章

随机推荐

  • java科学计数法转换为普通数字_Java BigDecimal toString() 的转换和输出

    BigDecimal 的 toString 方法将会把 BigDecimal 通过字符串的方式输出 这个方法将会在必要的时候使用指数进行输出 具体的转换步骤是按照下面的步骤进行转换的 BigDecimal的非标度值的绝对值用字符 0 到 9
  • linxu/torch/pytorch会自动安装 cuda

    如下图所示
  • Android 是否可支持 exFAT 格式 U 盘?如何使 Android 设备支持 exFAT

    Android 是否可支持 exFAT 格式 U 盘 如何使 Android 设备支持 exFAT exFAT 文件系统是一种适用于移动设备和嵌入式系统的文件系统 它具有更好的兼容性和更高的性能 然而 并非所有的 Android 设备都原生
  • 华为OD机试真题-完美走位/滑动窗口【2023Q1】

    输入一个长度为4的倍数的字符串 字符串中仅包含WASD四个字母 将这个字符串中的连续子串用同等长度的仅包含WASD的字符串替换 如果替换后整个字符串中WASD四个字母出现的频数相同 那么我们称替换后的字符串是 完美走位 求子串的最小长度 如
  • 2023华为OD机试真题【严格递增字符串】【2023.Q2】

    题目描述 定义字符串完全由 A 和 B 组成 当然也可以全是A 或全是 B 如果字符串从前往后都是以字典序排列的 那么我们称之为严格递增字符串 给出一个字符串s 允许修改字符串中的任意字符 即可以将任何的 A 修改成 B 也可以将任何的 B
  • dll signing issue

    1 Verify if a dll has been signed sn exe v module dll Scenario sometimes for security reasons a Net program requires all
  • 命令行_第二篇 Anaconda Prompt 命令提示符

    Anaconda 安装外接库过程 1 打开Anaconda Prompt 输入conda list 就会显示已经安装好的库 2 如果已安装库中没有所需的库 则需要手动进行安装 3 pip install 外接库 更新所需要的外接库 Anac
  • 用 h2 替换 hsql

    开发时用hsql做数据库 没什么问题 部署的时候 碰到数据量比较大的情况 就有点感觉慢了 这个时候 想替换方案时 考虑了几个可能的替代 一个是mysql 肯定没问题 但是部署麻烦 被否定了 其他的java数据库 可选的包括 mocki 这个
  • C语言学习

    为了更好的理解数据结构 开始重温C语言 以前学习过C语言 所以此笔记可能有些天马行空 不系统 挂个学习链接 C语言实例说明 解剖C语言 C语言教程 C语言网 dotcpp com 1B 1byte 字节 8bit 一字节存两位16进制数 以
  • 数据科学猫:强化学习的定义

    进击的橘子猫正式改名上线啦 我的CSDN主页 https blog csdn net Orange Spotty Cat 也欢迎大家搜索微信公众号 进击的橘子猫 我也会定期分享数据科学 Python 大数据 项目管理与PPT的相关知识 让我
  • DUT处理延迟 对Monitor采数和验证环境结束机制的影响分析

    1 问题背景 一句话描述 验证环境中 当激励完成发送时 由于DUT存在处理延迟 monitor在延迟一段时间后才能采集到DUT完整的输出 如何设计验证环境的结束机制 此处的验证环境结束机制 可以认为是main phase的结束控制 但并不单
  • 21. Merge Two Sorted Lists

    题目描述 将两个升序链表合并为一个新的 升序 链表并返回 新链表是通过拼接给定的两个链表的所有节点组成的 示例 1 外链图片转存失败 源站可能有防盗链机制 建议将图片保存下来直接上传 img 0VfjZ6Ct 1686493063120 i
  • STM32CubeMX工程配置说明

    一 STM32CubeMX配置 1 1 设置时钟 单片机的时钟 相当于人的心跳 只要单片机工作 必须要开启时钟 STM32单片机共有4个时钟来源 名称 缩写 频率 外部连接 功能 用途 特性 外部高速晶体振荡器 HSE 4 16MHz 4
  • Android手把手实战APP首页 下拉刷新 自动加载

    一 概述 作为一名三年Android开发经验的程序员 今天和大家一起实战一款APP的首页功能 这个首页在我们平时接触中还是很常见的 虽然页面简单 但是里面涉及的功能点还是挺多的 代码如有不足的还望各路同仁指点一二 页面中使用的开发库 整个首
  • 国家税务总局全国增值税发票查验平台网站js逆向分析及全逆向算法还原

    本文教程针对的是2021年7月2日时国税查验平台的js分析 其中版本号为V2 0 06 009 主要分析内容为key9和flwq39以及fplx这3个参数的算法 其中key9分为获取验证码阶段和查验阶段 算法有所区别 flwq39同理 教程
  • js-倒计时

    p p
  • Redis的底层数据结构

    1 Redis的五种数据类型及七种底层结构 键的类型只能为字符串 值支持五种数据类型 字符串 列表 集合 散列表 有序集合 对于Redis来讲 对于键值对来说 键总是字符串 值就是五个中的一个 所以我们只用关心值的类型 值有五种数据类型 S
  • 理解CPU/寄存器/内存之间的关系

    转自 https blog csdn net qq 27689785 article details 82975575 CPU 寄存器 内存 因为要了解多线程 自然少不了一些硬件知识的科普 我没有系统学习过硬件知识 仅仅是从书上以及网络上看
  • Node.js全栈开发笔记与心得

    highlight a11y dark 一 Node js 全栈开发资料 1 前端入门基础 慕课网HTML CSS入门 慕课网JS入门 javascript进阶篇 菜鸟教程html部分 菜鸟教程CSS部分 阮一峰js入门 阮一峰es6教程
  • 一道有关逆序对的算法题---归并、树状数组、线段树三种解法

    这阵子巨忙 呜呜呜 上周末打了第十四届蓝桥杯的校内赛 总的来说 题目是很简单的 不过其中的压轴题挺有意思 所以我打算来一发博客记录记录这道算法题 废话不多说了 上题 问题描述 小蓝有一个序列 a 1 a 2 a n 每次可以交换相邻的两个元