力扣第 387 场周赛第四题 将元素分配到两个数组中 II 二分查找,离散化,线段树

本文主要是介绍力扣第 387 场周赛第四题 将元素分配到两个数组中 II 二分查找,离散化,线段树,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Problem: 100246. 将元素分配到两个数组中 II

在力扣的题解

赛时没做出来,想了个排序,其实排序总假设最坏的情况即倒序,那肯定超时。当时想到线段树了,但是好久没练没搞出来QAQ,我的板子的线段树下标是从1开始的。
(新题题解力扣审的挺严,我随便写的被截了,所以乖乖算了次复杂度嘤嘤嘤~)

文章目录

  • 思路
  • 解题方法
    • 1.离散化
    • 2.线段树
  • 复杂度
  • Code

思路

我们可以开数组arr1,arr2来统计其中数字出现的数目(下标 i 代表数值,值arr[i]代表出现的次数),然后就可以通过区间和快速得到数组中大于某个数的数目。

因为数据是不断变化的,所以我们得使用树状数组或者线段树。

解题方法

1.离散化

由于数据范围是1~1e9,太大了,不能开1e9的数组。
但是数据量只有1e5,所以我们可以离散化来处理。

离散化后我们要再找这个数的位置,使用二分查找

复制一个数组tmp = nums,我们对tmp进行排序,即完成的离散化。
(由于要使用线段树,所以下标从1开始。)

2.线段树

然后就是通过线段树获取区间和,我们要的是严格大于当前数的数目,找到这个数的下标,下一个位置到数组末尾这段区间的和就是数目。(可能是最后一个数,下一个位置就越界了,特殊处理一下即可)

参考线段树板子:线段树板子

复杂度

时间复杂度:

O ( n l o g 2 n ) O(nlog_2n) O(nlog2n)

二分 l o g 2 n log_2n log2n,建树 n l o g 2 n nlog_2n nlog2n,区间加 l o g 2 n log_2n log2n,区间修改 l o g 2 n log_2n log2n,遍历数组n,各算法间互补干扰,取最大 n l o g 2 n nlog_2n nlog2n

空间复杂度:

O ( 2 n + n ) O(2^n+n) O(2n+n)

对于线段树数组的计算: 1 ∗ 2 ∗ . . . ∗ n 1*2*...*n 12...n,公比为2的等比数列, 1 ∗ 2 n − 1 1 1*\frac{2^{n}-1}{1} 112n1 即 2^n
再加上原数组大小n

Code

类ST是板子;
getmid是二分找下标;
tmp是离散化后下标对应的值的数组。

#define ll long long
class Solution {
public:template<class T>class ST//segment tree{struct node{T val;T t;//懒标记//服务后代node(T v = 0) :val(v), t(0){}};int n = a.size();vector<T>a;vector<node>d;public:void build_tree(int i, int l, int r){if (l == r){d[i].val = a[l];return;}int mid = l + (r - l) / 2;build_tree(i * 2, l, mid);build_tree(i * 2 + 1, mid + 1, r);d[i].val = d[i * 2].val + d[i * 2 + 1].val;}void spread(int i, int l, int r, int aiml, int aimr){int mid = l + (r - l) / 2;if (d[i].t != 0 && l != r){d[i * 2].val += d[i].t * (mid - l + 1);d[i * 2 + 1].val += d[i].t * (r - mid);d[i * 2].t += d[i].t;//可能上上次也没改d[i * 2 + 1].t += d[i].t;d[i].t = 0;}}T getsum(int l, int r){return _getsum(1, 1, n, l, r);}T _getsum(int i, int l, int r, int aiml, int aimr){if (aiml <= l && r <= aimr)//查询区间的子集全部加起来return d[i].val;//访问int mid = l + (r - l) / 2;spread(i, l, r, aiml, aimr);T ret = 0;if (aiml <= mid)ret += _getsum(i * 2, l, mid, aiml, aimr);if (aimr >= mid + 1)ret += _getsum(i * 2 + 1, mid + 1, r, aiml, aimr);return ret;}void update(int l, int r, ll val){_update(1, 1, n, l, r, val);//加并挂标记}void _update(int i, int l, int r, int aiml, int aimr, ll val){if (aiml <= l && r <= aimr){d[i].val += val * (r - l + 1);d[i].t += val;return;}int mid = l + (r - l) / 2;spread(i, l, r, aiml, aimr);if (aiml <= mid)_update(i * 2, l, mid, aiml, aimr, val);if (aimr >= mid + 1)_update(i * 2 + 1, mid + 1, r, aiml, aimr, val);//我们只对叶子更新了,(别多想懒标记)d[i].val = d[i * 2].val + d[i * 2 + 1].val;}ST(vector<T>arr){a = arr;n = a.size() - 1;d = vector<node>((ll)pow((ll)2, (ll)log2(n) + 1 + 1) + 10);build_tree(1, 1, n);}};//vector<int>tmp;int n;int getmid(int aim){int l = 0, r = n;while (l < r){int m = (l + r + 1) / 2;if (tmp[m] <= aim)l = m;else r = m - 1;}return l;}vector<int> resultArray(vector<int>& nums){n = nums.size();//tmp = nums;tmp = vector<int>(n + 1);for (int i = 1; i <= n; i++){tmp[i] = nums[i - 1];}sort(tmp.begin() + 1, tmp.end());vector<int>arr1, arr2, ret;vector<int>sarr1(n+1), sarr2(n+1);ST<int>demo1(sarr1), demo2(sarr2);arr1.push_back(nums[0]);arr2.push_back(nums[1]);int t = getmid(nums[0]);demo1.update(t, t, 1);t = getmid(nums[1]);demo2.update(t, t, 1);for (int i = 2; i < nums.size(); i++){t = getmid(nums[i]);//这个值的下标,我们求右边数目       if (t == n){if (arr1.size() <= arr2.size())arr1.push_back(nums[i]), demo1.update(t, t, 1);elsearr2.push_back(nums[i]), demo2.update(t, t, 1);continue;}int r1 = demo1.getsum(t + 1, n), r2 = demo2.getsum(t + 1, n);if (r1 == r2){if (arr1.size() <= arr2.size())arr1.push_back(nums[i]), demo1.update(t, t, 1);elsearr2.push_back(nums[i]), demo2.update(t, t, 1);}else if (r1 > r2)arr1.push_back(nums[i]), demo1.update(t, t, 1);elsearr2.push_back(nums[i]), demo2.update(t, t, 1);}ret = arr1;for (auto x : arr2)ret.push_back(x);return ret;}
};

这篇关于力扣第 387 场周赛第四题 将元素分配到两个数组中 II 二分查找,离散化,线段树的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/770828

相关文章

C++一个数组赋值给另一个数组方式

《C++一个数组赋值给另一个数组方式》文章介绍了三种在C++中将一个数组赋值给另一个数组的方法:使用循环逐个元素赋值、使用标准库函数std::copy或std::memcpy以及使用标准库容器,每种方... 目录C++一个数组赋值给另一个数组循环遍历赋值使用标准库中的函数 std::copy 或 std::

使用C++实现链表元素的反转

《使用C++实现链表元素的反转》反转链表是链表操作中一个经典的问题,也是面试中常见的考题,本文将从思路到实现一步步地讲解如何实现链表的反转,帮助初学者理解这一操作,我们将使用C++代码演示具体实现,同... 目录问题定义思路分析代码实现带头节点的链表代码讲解其他实现方式时间和空间复杂度分析总结问题定义给定

C++初始化数组的几种常见方法(简单易懂)

《C++初始化数组的几种常见方法(简单易懂)》本文介绍了C++中数组的初始化方法,包括一维数组和二维数组的初始化,以及用new动态初始化数组,在C++11及以上版本中,还提供了使用std::array... 目录1、初始化一维数组1.1、使用列表初始化(推荐方式)1.2、初始化部分列表1.3、使用std::

C++ Primer 多维数组的使用

《C++Primer多维数组的使用》本文主要介绍了多维数组在C++语言中的定义、初始化、下标引用以及使用范围for语句处理多维数组的方法,具有一定的参考价值,感兴趣的可以了解一下... 目录多维数组多维数组的初始化多维数组的下标引用使用范围for语句处理多维数组指针和多维数组多维数组严格来说,C++语言没

Python如何计算两个不同类型列表的相似度

《Python如何计算两个不同类型列表的相似度》在编程中,经常需要比较两个列表的相似度,尤其是当这两个列表包含不同类型的元素时,下面小编就来讲讲如何使用Python计算两个不同类型列表的相似度吧... 目录摘要引言数字类型相似度欧几里得距离曼哈顿距离字符串类型相似度Levenshtein距离Jaccard相

使用Navicat工具比对两个数据库所有表结构的差异案例详解

《使用Navicat工具比对两个数据库所有表结构的差异案例详解》:本文主要介绍如何使用Navicat工具对比两个数据库test_old和test_new,并生成相应的DDLSQL语句,以便将te... 目录概要案例一、如图两个数据库test_old和test_new进行比较:二、开始比较总结概要公司存在多

CSS3中使用flex和grid实现等高元素布局的示例代码

《CSS3中使用flex和grid实现等高元素布局的示例代码》:本文主要介绍了使用CSS3中的Flexbox和Grid布局实现等高元素布局的方法,通过简单的两列实现、每行放置3列以及全部代码的展示,展示了这两种布局方式的实现细节和效果,详细内容请阅读本文,希望能对你有所帮助... 过往的实现方法是使用浮动加

C#比较两个List集合内容是否相同的几种方法

《C#比较两个List集合内容是否相同的几种方法》本文详细介绍了在C#中比较两个List集合内容是否相同的方法,包括非自定义类和自定义类的元素比较,对于非自定义类,可以使用SequenceEqual、... 目录 一、非自定义类的元素比较1. 使用 SequenceEqual 方法(顺序和内容都相等)2.

Java 字符数组转字符串的常用方法

《Java字符数组转字符串的常用方法》文章总结了在Java中将字符数组转换为字符串的几种常用方法,包括使用String构造函数、String.valueOf()方法、StringBuilder以及A... 目录1. 使用String构造函数1.1 基本转换方法1.2 注意事项2. 使用String.valu

在MyBatis的XML映射文件中<trim>元素所有场景下的完整使用示例代码

《在MyBatis的XML映射文件中<trim>元素所有场景下的完整使用示例代码》在MyBatis的XML映射文件中,trim元素用于动态添加SQL语句的一部分,处理前缀、后缀及多余的逗号或连接符,示... 在MyBATis的XML映射文件中,<trim>元素用于动态地添加SQL语句的一部分,例如SET或W