2023-12-09:用go语言,给你两个整数数组 arr1 和 arr2, 返回使 arr1 严格递增所需要的最小「操作」数(可能为 0)。 每一步「操作」中,你可以分别从 arr1 和 arr2
2023-12-09:用go语言,给你两个整数数组 arr1 和 arr2,
返回使 arr1 严格递增所需要的最小「操作」数(可能为 0)。
每一步「操作」中,你可以分别从 arr1 和 arr2 中各选出一个索引,
分别为 i 和 j,0 <= i < arr1.length 和 0 <= j < arr2.length,
然后进行赋值运算 arr1[i] = arr2[j]。如果无法让 arr1 严格递增,请返回 -1。
输入:arr1 = [1,5,3,6,7], arr2 = [4,3,1]。
输出:2。
答案2023-12-09:
大体过程如下:
算法1(makeArrayIncreasing1):
1.对arr2进行排序并去除重复元素,生成新的数组help,并统计cnt为help的长度。
2.通过递归函数process1来计算从arr1的索引i开始到结尾的最小操作数。初始时,i为-1。
3.在process1中,通过二分查找函数find,在arr2中找到第一个大于cur的元素的索引f。
4.使用循环遍历arr1中从i+1到末尾的元素。
若当前元素大于cur,则说明可以选择该元素,继续递归调用process1,并将操作数times加1。
若f不等于-1且小于arr2的长度,更新cur为arr2[f],同时f加1,times加1。
若f等于-1或大于等于arr2的长度,跳出循环。
5.返回递归调用的结果ans,即最小操作数。
算法2(makeArrayIncreasing2):
1.对arr2进行排序并去除重复元素,生成新的数组help,并统计cnt为help的长度。
2.创建dp数组,初始值为-1。
3.通过递归函数process2来计算从arr1的索引i开始到结尾的最小操作数。同时,使用dp数组记录已计算过的结果,以便后续查询。
4.在process2中,若dp[i+1]不等于-1,直接返回dp[i+1]。
5.剩下的过程与makeArrayIncreasing1基本相同,只是将递归调用替换为对dp数组的查询和更新。
6.返回递归调用的结果ans,同时将其保存到dp[i+1]中。
算法3(makeArrayIncreasing3):
1.对arr2进行排序并去除重复元素,生成新的数组arr2,并统计m为arr2的长度。
2.创建dp数组,长度为n+2,并初始化为最大整数。
3.从arr1末尾向前遍历,使用循环计算从索引i开始到结尾的最小操作数。
初始化cur为arr1[i],f为在arr2中找到第一个大于cur的元素的索引。
初始化dp[i+1]为最大整数,times为0。
使用循环遍历arr1中从i+1到末尾的元素,操作步骤与makeArrayIncreasing1和makeArrayIncreasing2相似。
若dp[j+1]不等于最大整数,更新dp[i+1]为times+dp[j+1]与dp[i+1]中的较小值。
若f不等于-1且小于m,更新cur为arr2[f],同时f加1,times加1。
若f等于-1或大于等于m,跳出循环。
4.若dp[0]等于最大整数,返回-1;否则返回dp[0]作为最小操作数。
时间复杂度分析:
算法1和算法2的时间复杂度为O(n * m),其中n和m分别为arr1和arr2的长度,因为每个元素都需要遍历一次。
算法3的时间复杂度为O(n * m + m * log(m)),其中m为arr2的长度,因为要对arr2进行排序并进行二分查找。
额外空间复杂度分析:
算法1和算法2的额外空间复杂度为O(m),用于存储去重后的arr2。
算法3的额外空间复杂度为O(m),用于存储去重后的arr2,并且使用了一个大小为n+2的dp数组来记录中间结果。
go完整代码如下:
package main
import (
"fmt"
"math"
"sort"
)
func makeArrayIncreasing1(arr1 []int, arr2 []int) int {
sort.Ints(arr2)
cnt := 1
for i := 1; i < len(arr2); i++ {
if arr2[i] != arr2[i-1] {
cnt++
}
}
help := make([]int, cnt)
help[0] = arr2[0]
for i, j := 1, 1; i < len(arr2); i++ {
if arr2[i] != arr2[i-1] {
help[j] = arr2[i]
j++
}
}
ans := process1(arr1, help, -1)
if ans == math.MaxInt64 {
return -1
}
return ans
}
func process1(arr1 []int, arr2 []int, i int) int {
if i == len(arr1) {
return 0
} else {
cur := 0
if i == -1 {
cur = math.MinInt64
} else {
cur = arr1[i]
}
f := find(arr2, cur)
ans := math.MaxInt64
times := 0
for j := i + 1; j <= len(arr1); j++ {
if j == len(arr1) || cur < arr1[j] {
next := process1(arr1, arr2, j)
if next != math.MaxInt64 {
ans = min(ans, times+next)
}
}
if f != -1 && f < len(arr2) {
cur = arr2[f]
f++
times++
} else {
break
}
}
return ans
}
}
func find(arr2 []int, num int) int {
l := 0
r := len(arr2) - 1
ans := -1
for l <= r {
m := (l + r) / 2
if arr2[m] > num {
ans = m
r = m - 1
} else {
l = m + 1
}
}
return ans
}
func min(a, b int) int {
if a < b {
return a
}
return b
}
func makeArrayIncreasing2(arr1 []int, arr2 []int) int {
sort.Ints(arr2)
cnt := 1
for i := 1; i < len(arr2); i++ {
if arr2[i] != arr2[i-1] {
cnt++
}
}
help := make([]int, cnt)
help[0] = arr2[0]
for i, j := 1, 1; i < len(arr2); i++ {
if arr2[i] != arr2[i-1] {
help[j] = arr2[i]
j++
}
}
dp := make([]int, len(arr1)+1)
for i := range dp {
dp[i] = -1
}
ans := process2(arr1, help, -1, dp)
if ans == math.MaxInt64 {
return -1
}
return ans
}
func process2(arr1 []int, arr2 []int, i int, dp []int) int {
if i == len(arr1) {
return 0
} else {
if dp[i+1] != -1 {
return dp[i+1]
}
cur := 0
if i == -1 {
cur = math.MinInt64
} else {
cur = arr1[i]
}
f := find(arr2, cur)
ans := math.MaxInt64
times := 0
for j := i + 1; j <= len(arr1); j++ {
if j == len(arr1) || cur < arr1[j] {
next := process2(arr1, arr2, j, dp)
if next != math.MaxInt64 {
ans = min(ans, times+next)
}
}
if f != -1 && f < len(arr2) {
cur = arr2[f]
f++
times++
} else {
break
}
}
dp[i+1] = ans
return ans
}
}
func makeArrayIncreasing3(arr1 []int, arr2 []int) int {
sort.Ints(arr2)
m := 1
for i := 1; i < len(arr2); i++ {
if arr2[i] != arr2[m-1] {
arr2[m] = arr2[i]
m++
}
}
n := len(arr1)
dp := make([]int, n+2)
for i := n - 1; i >= -1; i-- {
cur := 0
if i == -1 {
cur = math.MinInt64
} else {
cur = arr1[i]
}
f := find3(arr2, m, cur)
dp[i+1] = math.MaxInt64
times := 0
for j := i + 1; j <= n; j++ {
if j == n || cur < arr1[j] {
if dp[j+1] != math.MaxInt64 {
dp[i+1] = min(dp[i+1], times+dp[j+1])
}
}
if f != -1 && f < m {
cur = arr2[f]
f++
times++
} else {
break
}
}
}
if dp[0] == int(^uint(0)>>1) {
return -1
}
return dp[0]
}
func find3(arr2 []int, size int, num int) int {
l := 0
r := size - 1
ans := -1
for l <= r {
m := (l + r) / 2
if arr2[m] > num {
ans = m
r = m - 1
} else {
l = m + 1
}
}
return ans
}
func main() {
if true {
arr1 := []int{1, 5, 3, 6, 7}
arr2 := []int{4, 3, 1}
fmt.Println(makeArrayIncreasing1(arr1, arr2))
}
if true {
arr1 := []int{1, 5, 3, 6, 7}
arr2 := []int{4, 3, 1}
fmt.Println(makeArrayIncreasing2(arr1, arr2))
}
if true {
arr1 := []int{1, 5, 3, 6, 7}
arr2 := []int{4, 3, 1}
fmt.Println(makeArrayIncreasing3(arr1, arr2))
}
}

2023-12-09:用go语言,给你两个整数数组 arr1 和 arr2, 返回使 arr1 严格递增所需要的最小「操作」数(可能为 0)。 每一步「操作」中,你可以分别从 arr1 和 arr2的更多相关文章
- c语言经典算法——查找一个整数数组中第二大数
题目: 实现一个函数,查找一个整数数组中第二大数. 算法思想: 设置两个变量max1和max2,用来保存最大数和第二大数,然后将数组剩余的数依次与这两个数比较,如果这个数a比max1大,则先将max1 ...
- Leecode刷题之旅-C语言/python-88合并两个有序数组
/* * @lc app=leetcode.cn id=88 lang=c * * [88] 合并两个有序数组 * * https://leetcode-cn.com/problems/merge-s ...
- 12天学好C语言——记录我的C语言学习之路(Day 12)
12天学好C语言--记录我的C语言学习之路 Day 12: 进入最后一天的学习,用这样一个程序来综合考量指针和字符串的关系,写完这个程序,你对字符串和指针的理解应该就不错了. //输入一个字符串,内有 ...
- 12天学好C语言——记录我的C语言学习之路(Day 11)
12天学好C语言--记录我的C语言学习之路 Day 11: 因为指针部分比较的难,所以我们花费的时间也是最长的,希望大家耐的住性子,多多理解,多多打代码.好了,废话不多说,来看第11天的学习. //编 ...
- 12天学好C语言——记录我的C语言学习之路(Day 9)
12天学好C语言--记录我的C语言学习之路 Day 9: 函数部分告一段落,但是我们并不是把函数完全放下,因为函数无处不在,我们今后的程序仍然会大量运用到函数 //转入指针部分的学习,了解指针是什么 ...
- 12天学好C语言——记录我的C语言学习之路(Day 8)
12天学好C语言--记录我的C语言学习之路 Day 8: 从今天开始,我们获得了C语言中很有力的一个工具,那就是函数.函数的魅力不仅于此,一个程序到最后都是由众多函数组成的,我们一定要用好函数,用熟练 ...
- 12天学好C语言——记录我的C语言学习之路(Day 7)
12天学好C语言--记录我的C语言学习之路 Day 7: 昨天进行了一天的数组学习,今天大家可以先写几个昨天的程序热热身,回顾回顾,然后今天第一个新程序也是关于数组的,比较难,准备好就开始啦! //输 ...
- 12天学好C语言——记录我的C语言学习之路(Day 3)
12天学好C语言--记录我的C语言学习之路 Day 3: 不知不觉到了第三天的学习,我们前两天学习的东西很杂乱,各个方面都有学习.我觉得这不是不系统,也不是学的不扎实,这种学习对于初学者而言我认为是很 ...
- 12天学好C语言——记录我的C语言学习之路(Day 1)
12天学好C语言--记录我的C语言学习之路 Day 1: 刚刚入门C语言,那么肯定要先把什么是C语言和大家讲清楚,那么大家看下面一段程序(program 1.1): /*//program 1.1 ...
- 12天学好C语言——记录我的C语言学习之路(Day 10)
12天学好C语言--记录我的C语言学习之路 Day 10: 接着昨天的指针部分学习,有这么一个题目: //还是四个学生,四门成绩,只要有学生一门功课没及格就输出这个学生的所有成绩 /*//progra ...
随机推荐
- Python 遍历字典的若干方法
哈喽大家好,我是咸鱼 我们知道字典是 Python 中最重要且最有用的内置数据结构之一,它们无处不在,是语言本身的基本组成部分 我们可以使用字典来解决许多编程问题,那么今天我们就来看看如何在 Pyth ...
- Golang日志新选择:slog
go1.21中,slog这一被Go语言团队精心设计的结构化日志包正式落地,本文将带领读者上手slog,体会其与传统log的差异. WHY 在日志处理上,我们从前使用的log包缺乏结构化的输出,导致信息 ...
- 20个最佳实践提升Terraform工作流程|Part 1
Terraform 是管理基础设施及代码(IaC)最常用的工具之一,它能使我们安全且可预测地对基础设施应用更改.刚开始上手 Terraform 可能会感觉有些不容易,但很快就能对该工具有基本的了解,随 ...
- SpringBoot 后端配置 Https 教程
以阿里云为例子 1. 申请 SSL 证书 1. 注册域名 打开阿里云官网,搜索域名 点击域名注册,输入域名,点击搜索 选择心仪的域名,点击购买,打钱 进入域名控制台,进行实名认证 2. 申请 SSL ...
- fmt 库简介和示例【GO 基础】
〇.关于 fmt fmt 标准库是 Go 语言标准库的一部分,提供了格式化字符串.输入输出等基本功能.通过 fmt 库,我们可以进行简单的格式化输出.读取用户输入.错误输出等操作. fmt 库实现了类 ...
- tensorboard可视化点云
tensorboard可视化点云 用 tensorboard 自带的 add_mesh 方法:支持可视化点云和网格,参考链接: tensorflow 网站的tensorboard pytorch 网站 ...
- 2023-10-07:用go语言,给定n个二维坐标,表示在二维平面的n个点, 坐标为double类型,精度最多小数点后两位, 希望在二维平面上画一个圆,圈住其中的k个点,其他的n-k个点都要在圆外。
2023-10-07:用go语言,给定n个二维坐标,表示在二维平面的n个点, 坐标为double类型,精度最多小数点后两位, 希望在二维平面上画一个圆,圈住其中的k个点,其他的n-k个点都要在圆外. ...
- Sunshine on my shoulders
https://music.163.com/#/song?id=1477706 Sunshine on my shoulders makes me happy照在我肩上的阳光让我欢乐Sunshine ...
- 单元测验3:亲密关系mooc
单元测验3:亲密关系 查看帮助 返回 1 单选(2分) 在亲密关系中,有关权力的表述,以下说法不太准确的的是? A. 对关系付出越多,权力越大. B. 大部分人会倾向认为,在恋爱关系中,男女应该拥 ...
- 14.8 Socket 一收一发通信
通常情况下我们在编写套接字通信程序时都会实现一收一发的通信模式,当客户端发送数据到服务端后,我们希望服务端处理请求后同样返回给我们一个状态值,并以此判断我们的请求是否被执行成功了,另外增加收发同步有助 ...