LeetCode 周赛 351(2023/06/25)T2 有点意思

news2025/1/12 10:53:10

本文已收录到 AndroidFamily,技术和职场问题,请关注公众号 [彭旭锐] 和 [BaguTree Pro] 知识星球提问。

  • 往期回顾:LeetCode 单周赛第 348 场 · 数位 DP 模版学会了吗?

T1. 美丽下标对的数目(Easy)

  • 标签:计数 + 数学

T2. 得到整数零需要执行的最少操作数(Medium)

  • 标签:数学

T3. 得到整数零需要执行的最少操作数(Medium)

  • 标签:乘法原理

T4. 机器人碰撞(Hard)

  • 标签:栈


T1. 美丽下标对的数目(Easy)

https://leetcode.cn/problems/number-of-beautiful-pairs/

题解一(暴力)

两层扫描,同时检查前驱中匹配的配对数。

class Solution {
    fun countBeautifulPairs(nums: IntArray): Int {
        var ret = 0
        for (i in nums.indices) {
            var x = nums[i]
            while (x >= 10) x /= 10
            for (j in i + 1 until nums.size) {
                if (gcb(nums[j] % 10, x) == 1) ret++
            }
        }
        return ret
    }

    private fun gcb(x: Int, y: Int) : Int {
        var a = x
        var b = y
        while (b != 0) {
            val temp = a % b
            a = b
            b = temp
        }
        return a
    }
}

复杂度分析:

  • 时间复杂度: O ( n 2 ) O(n^2) O(n2)
  • 空间复杂度: O ( 1 ) O(1) O(1)

题解二(计数 + 数学)

线性扫描数组,同时检查前驱中匹配的配对数。由于题目只考虑前驱数字的最高位和当前位置的最低位,我们可以维护前驱数字的最高位出现次数。

class Solution {
    fun countBeautifulPairs(nums: IntArray): Int {
        var ret = 0
        val cnt = IntArray(10)
        for (i in nums.indices) {
            for (j in 1 .. 9) {
                if (cnt[j] > 0 && gcb(nums[i] % 10, j) == 1) ret += cnt[j]
            }
            var x = nums[i]
            while (x >= 10) x /= 10
            cnt[x]++
        }
        return ret
    }

    private fun gcb(x: Int, y: Int) : Int {
        var a = x
        var b = y
        while (b != 0) {
            val temp = a % b
            a = b
            b = temp
        }
        return a
    }
}

复杂度分析:

  • 时间复杂度: O ( C ⋅ n ) O(C·n) O(Cn) 其中 C = 10;
  • 空间复杂度: O ( C ) O(C) O(C)

T2. 得到整数零需要执行的最少操作数(Medium)

https://leetcode.cn/problems/minimum-operations-to-make-the-integer-zero/

这道题的思维难度比较高。

同时考虑 2^i 和 nums2 不好处理,我们可以尝试分别处理:观察示例 1(最小操作次数为 3),如果我们先对 num1 减去 3 次 nums2,则得到二进制 1101,正好可以通过减去 3 次 2^i 清零(-1、-4 和 -8)。

// 0011 + 2
// => 0101 + 2
// => 0111 + 2
// => 1101 (-1 - 4 - 8)

因此,我们假设操作 k 次后可以消除 num1,那么需要有 nums1 - knum2 的二进制位正好存在 k 个 1,此时就可以用 k 次 2^i 消除。那么我们的问题就转换为是否存在 k,使得 nums1 - knums2 的二进制位中 1 的个数为 k。

if (k == (nums1 - k * nums2).bitCount()) return true

然而,这个思路是有陷阱的,比如说操作 4 次后的二进制位中 1 的个数只有 3 个,按照上面的思路是非法的,但事实上我们依然可以通过操作 4 次来清零(-1、-4、-8 ⇒ 将 -8 拆分为 2 次 -4,总的操作次数就是 -1、-4、-4、-4);

  • 最少操作次数:每次将二进制位中的 1 消除;
  • 最多操作次数:每次减 1。

综上所述,令 x 为 num1 - k * num2,y 为 x 二进制位中 1 的个数,从 1 开始枚举 k,那么当满足 y ≤ k 且 x ≥ k 时,必然可以通过 k 次操作清零。

// 0001 + 2
// => 0011 + 2
// => 0101 + 2
// => 0111 + 2
// => 1101

最后一个问题,复杂度怎么算,显然取决于 k 的上界:

  • 当 num2 == 0 时,操作次数直接等于 num1 二进制位中 1 的个数,最大操作次数是 log(num1);
  • 当 num2 > 0 或 num2 < 0 时,算法在 k ≥ bitCount(x) 时终止,最大操作次数是 log(x)。
class Solution {
    fun makeTheIntegerZero(num1: Int, num2: Int): Int {
        var k = 1
        while (true) {
            val x = num1 - 1L * k * num2
            if (k > x) return -1
            if (k >= java.lang.Long.bitCount(x)) return k
            k++
        }
    }
}
class Solution {
    fun makeTheIntegerZero(num1: Int, num2: Int): Int {
        var k = 1
        var x = 1L * num1
        while (true) {
            x -= num2
            if (k > x) return -1
            if (k >= java.lang.Long.bitCount(x)) return k
            k++
        }
    }
}

复杂度分析:

  • 时间复杂度: O ( l g x ) O(lgx) O(lgx)
  • 空间复杂度: O ( 1 ) O(1) O(1)

T3. 得到整数零需要执行的最少操作数(Medium)

https://leetcode.cn/problems/ways-to-split-array-into-good-subarrays/

题解(分组 + 乘法原理)

以数字 1 为分割线,将每段连续的 0 分为一组,再用乘法原理计算总方案数。

class Solution {
    fun numberOfGoodSubarraySplits(nums: IntArray): Int {
        // 分组 + 乘法原理
        val MOD = 1000000007
        var ret = 1L
        var pre1 = -1
        for ((i, num) in nums.withIndex()) {
            if (num == 0) continue
            if (pre1 != -1) ret = ret * (i - pre1) % MOD
            pre1 = i
        }
        return if (pre1 == -1) 0 else ret.toInt()
    }
}

复杂度分析:

  • 时间复杂度: O ( n ) O(n) O(n)
  • 空间复杂度: O ( 1 ) O(1) O(1)

T4. 机器人碰撞(Hard)

https://leetcode.cn/problems/robot-collisions/

题解(栈)

这道题与经典题 735. 行星碰撞 几乎是一样的。

我们使用栈模拟保留的机器人,枚举机器人,当机器人与栈顶方向冲突时按规则消除,最后输出栈内剩余的机器人。

class Solution {
    fun survivedRobotsHealths(positions: IntArray, healths: IntArray, directions: String): List<Int> {
        // 排序
        val indexs = Array(positions.size) { it }
        Arrays.sort(indexs) { i1, i2 ->
            positions[i1] - positions[i2]
        }
        // 模拟 <index>
        val stack = ArrayDeque<Int>()
        outer@ for (id in indexs) {
            // 当前机器人向右,不会发生碰撞
            if (directions[id] == 'R') {
                stack.push(id)
                continue
            }
            while (!stack.isEmpty() && directions[stack.peek()] == 'R') {
                var topId = stack.peek()
                if (healths[topId] > healths[id]) {
                    // 栈顶健康度 -1
                    if (--healths[topId] == 0) stack.poll()
                    continue@outer
                } else if(healths[topId] < healths[id]) {
                    // 弹出栈顶
                    healths[id] -= 1
                    stack.poll()
                } else {
                    // 弹出栈顶
                    stack.poll()
                    continue@outer
                }
            }
            if (healths[id] > 0) stack.push(id)
            // println(stack.joinToString())
        }
        // 输出
        val ret = stack.toMutableList()
        ret.sort() // 题目要求按照原位置顺序输出
        for (i in ret.indices) {
            ret[i] = healths[ret[i]]
        }
        return ret
    }
}

复杂度分析:

  • 时间复杂度: O ( n l g n ) O(nlgn) O(nlgn) 瓶颈在排序上;
  • 空间复杂度: O ( n ) O(n) O(n) 栈空间。

往期回顾

  • LeetCode 单周赛第 348 场 · 数位 DP 模版学会了吗?
  • LeetCode 单周赛第 347 场 · 二维空间上的 LIS 最长递增子序列问题
  • LeetCode 双周赛第 104 场 · 流水的动态规划,铁打的结构化思考
  • LeetCode 双周赛第 103 场 · 区间求和的树状数组经典应用

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/685160.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

python爬虫并做可视化分析--前程无忧

一.数据采集 1.采集逻辑 2.数据schema 招聘信息Schema { "岗位名称": "财务会计主管", "薪资":"1.3-2万", "地址": "*******", "经验要求": "5-7年", "公司名": "***…

JDK8新特性-上部

文章目录 一、Java发展史1.1 发展史1.2 OpenJDK和OracleJDK1.3 Open JDK 官网介绍 二、Lambda表达式2.1 需求分析2.2 Lamada表达式的体验2.3 Lambda表达式的语法规则2.3.1 Lambda表达式练习2.3.2 Lambda表达式练习 2.4 Lambda表达式的使用前提2.5 FunctionalInterface注解2.6 L…

MATLAB | 如何使用MATLAB获取顶刊《PNAS》绘图(附带近3年图像)

千呼万唤始出来&#xff0c;《PNAS》绘图获取的代码来啦&#xff0c;不过这次研究了半天也没想到如何获取付费文章的绘图&#xff0c;就只下载了免费文章(主要也怕侵权)&#xff0c;不过光免费文章的图片三年了也有接近1.7w张了&#xff0c;同时使用代码下载时依旧需要科学上网…

【Redis】Redis的数据结构

【Redis】Redis的数据结构 文章目录 【Redis】Redis的数据结构1. 动态字符串SDS2. IntSet2.1 IntSet升级 3. Dict3.1 Dict的扩容3.2 Dict的收缩3.3 Dict的rehash 4. ZipList4.1 ZipList中的Entry4.1.1 Encoding编码 4.2 ZipList的连锁更新问题4.3 特性 5. QuickList 1. 动态字符…

【软考网络管理员】2023年软考网管初级常见知识考点(7)-生成树协议

涉及知识点 STP的原理&#xff0c;端口的状态&#xff0c;RSTP协议&#xff0c;MSTP协议&#xff0c;软考网络管理员常考知识点&#xff0c;软考网络管理员网络安全&#xff0c;网络管理员考点汇总。 原创于&#xff1a;CSDN博主-《拄杖盲学轻声码》&#xff0c;更多考点汇总可…

模拟电路系列分享-运放的关键参数2

目录 概要 整体架构流程 技术名词解释 1.输入偏置电流&#xff1a; 2.输入失调电流 技术细节 总结; 概要 提示&#xff1a;这里可以添加技术概要 实际运放与理想运放具有很多差别。理想运放就像一个十全十美的人&#xff0c;他学习100 分&#xff0c;寿命无限长&#xff0c;长…

Modal对话框(antd-design组件库)展示所有配置选项和onChange的作用

1.Modal对话框 模态对话框。 2.何时使用 需要用户处理事务&#xff0c;又不希望跳转页面以致打断工作流程时&#xff0c;可以使用 Modal 在当前页面正中打开一个浮层&#xff0c;承载相应的操作。 另外当需要一个简洁的确认框询问用户时&#xff0c;可以使用 App.useApp 封装的…

创建微信小程序的几种方式

创建微信小程序的几种方式 1. 使用原生方式 在官网上下载微信开发者工具&#xff0c;之后使用微信开发者工具新建项目即可。 微信这边提供了多个模板&#xff0c;可以直接下载模板快速搭建上线&#xff0c;也可以使用空白模板根据需求自行编写。 空白模板项目结构&#xff1…

C语言:打印0-100000中的自幂数(水仙花数是其中一种)

题目&#xff1a; 求出 0&#xff5e;100000 之间的所有 自幂数 并输出。 自幂数是指一个n位数&#xff0c;其各位数字的n次方之和恰好等于该数本身&#xff0c; 如:153&#xff1d;1^3&#xff0b;5^3&#xff0b;3^3&#xff0c; 则153是一个自幂数。 思路&#xff1a; 总体…

【数据网格架构】分布式数据网格作为集中式数据单体的解决方案

企业数据架构师不应构建大型集中式数据平台&#xff0c;而应创建分布式数据网格。 ThoughtWorks 的首席技术顾问 Zhamak Dehghani 在旧金山 QCon 的演讲和相关文章中表示&#xff0c;这种方法的改变需要范式转变。随着数据变得越来越普遍&#xff0c;传统的数据仓库和数据湖架构…

Linux基础+命令操作+mysql、tomcat、nginx、RabbitMQ、Redis,ElasticSearch

配置代理 一、永久设置 //编辑配置文件 vi /etc/profile //在该配置文件的最后添加代理配置 export http_proxyhttp://f1336515:password10.137.255.169:3128 //代理服务器ip地址和端口号 export https_proxyhttp://f1336515:password10.137.255.169:3128 //代理服务器ip…

【软考网络管理员】2023年软考网管初级常见知识考点(11)-TCP和UDP详解

涉及知识点 传输控制协议TCP是什么&#xff0c;三次握手的概念理解&#xff0c;用户数据报协议UDP是什么&#xff0c;软考网络管理员常考知识点&#xff0c;软考网络管理员网络安全&#xff0c;网络管理员考点汇总。 原创于&#xff1a;CSDN博主-《拄杖盲学轻声码》&#xff0…

AntV G6新版源码浅析

前言 AntV是蚂蚁金服全新一代数据可视化解决方案&#xff0c;其中G6主要用于解决图可视领域相关的前端可视化问题&#xff0c;其是一个简单、易用、完备的图可视化引擎。本文旨在通过简要分析G6 5.x版本源码来对图可视领域的一些底层引擎进行一个大致了解&#xff0c;同时也为…

【玩转Linux操作】详细讲解expr,read,echo,printf,test,[]等命令

&#x1f38a;专栏【玩转Linux操作】 &#x1f354;喜欢的诗句&#xff1a;更喜岷山千里雪 三军过后尽开颜。 &#x1f386;音乐分享【free loop】 大一同学小吉&#xff0c;欢迎并且感谢大家指出我的问题&#x1f970; 文章目录 &#x1f354;expr命令⭐表达式说明 &#x1f3…

JAVA:Springboot动态装配Druid多数据源

1、简介 最近打算搭建一个鉴权中心服务&#xff0c;采用springbootFastMybatis装配Druid&#xff0c;考虑后续拓展采用Druid多数据源配置&#xff0c;以一个数据源为主&#xff0c;多个动态数据源为辅的结构。除了数据库&#xff0c;后续会结合shiro安全框架来搭建。 2、引用…

【Leetcode60天带刷】day33回溯算法——1005.K次取反后最大化的数组和 134. 加油站 135. 分发糖果

​ 题目&#xff1a; 1005. K 次取反后最大化的数组和 给你一个整数数组 nums 和一个整数 k &#xff0c;按以下方法修改该数组&#xff1a; 选择某个下标 i 并将 nums[i] 替换为 -nums[i] 。 重复这个过程恰好 k 次。可以多次选择同一个下标 i 。 以这种方式修改数组后&am…

将视频转为幻灯片图像:利用OpenCV实现视频资料转换的指南

视频成为了传播知识和信息的重要媒介之一。然而&#xff0c;有时我们需要以静态的形式保存视频内容&#xff0c;例如将视频讲座转换为幻灯片或图像&#xff0c;以便于分享、存档或打印。幸运的是&#xff0c;OpenCV这一功能强大的计算机视觉库提供了各种技术和工具&#xff0c;…

机器学习之线性回归算法

目录 线性回归算法 求导法推导 梯度下降法推导 线性回归实现人脸识别 导入数据 构建标签矩阵 经典线性回归求导法实现 经典线性回归梯度下降法实现 岭回归实现 套索回归实现 局部加权线性回归实现 可视化 人脸识别 线性回归算法 求导法推导 梯度下降法推导 线性回…

chatgpt赋能python:Title:Python编程中的空格怎么用?详细教程!

Title: Python编程中的空格怎么用&#xff1f;详细教程&#xff01; Introduction: Python编程的空格使用一直是令人困惑的话题之一&#xff0c;但它却是Python语言中非常重要的一部分。空格在Python程序中用来表示代码块的开始和结束&#xff0c;因此不同的空格使用方式可能…

【夜深人静学数据结构与算法 | 第十篇】动态规划

目录 前言&#xff1a; 动态规划&#xff1a; 常见应用&#xff1a; 解题步骤&#xff1a; 动态规划的简化步骤&#xff1a; 案例&#xff1a; 509. 斐波那契数 - 力扣&#xff08;LeetCode&#xff09; 70. 爬楼梯 - 力扣&#xff08;LeetCode&#xff09; 62. 不同路…