LeetCode——数组 移除元素(Java)

news2025/1/22 19:01:24

移除元素

  • 简介
  • [简单] 27. 移除元素
  • [简单] 26. 删除有序数组中的重复项
  • [简单] 283. 移动零
  • [简单] 844. 比较含退格的字符串
  • [简单] 977. 有序数组的平方

简介

记录一下自己刷题的历程以及代码。写题过程中参考了 代码随想录。会附上一些个人的思路,如果有错误,可以在评论区提醒一下。
一旦设计到数组移除元素,就可以首先考虑一下双指针法解题。快慢指针法经常可以比较高效的对数组做一遍处理,把需要删除的元素删掉进行压缩。

[简单] 27. 移除元素

原题链接

(注意:这样的方法是一种不保留原先顺序的方法)
方法①:其实也是一种双指针的思路。设置一个下标指向数组最右边,从头开始遍历,一旦遍历到有元素需要移除,就把他往最后放,并将下标左移,右边区域不再参与遍历,记住一旦有元素被置换到后面,需要将当前循环下标做i--处理,因为你换过来的元素依然可能是一个需要移除的元素。

class Solution {
    public int removeElement(int[] nums, int val) {
        int length = nums.length;
        int right = nums.length - 1; //替换指针
        for(int i = 0; i < length; i++){
            if(nums[i] == val){
                int temp = nums[right];
                nums[right] = nums[i];
                nums[i] = temp;
                right--;
                length--;  //相当于舍弃末尾的部分长度
                i--; //置换之后当前位置可能还是一个需要置换的元素,继续检查
            }
        }
        return length;
    }
}

(注意:该方法保留了原数组的顺序)
方法②:快慢指针法,相当于慢指针负责对快指针指向的元素进行复制,而快指针则会跳过那些不需要复制的元素。

class Solution {
    public int removeElement(int[] nums, int val) {
        int fastIndex = 0;
        int slowIndex = 0;
        int count = 0;
        while(fastIndex < nums.length && slowIndex < nums.length - count){
            nums[slowIndex] = nums[fastIndex];
            if(nums[fastIndex] == val){
                count++; 
            }else{
                slowIndex++;
            }
            fastIndex++;
        }
        return nums.length - count;
    }
}

[简单] 26. 删除有序数组中的重复项

原题链接

方法①:根据有序数组的条件,对上面的双指针法进行一定的改造,定义一个tag标记目前碰到的数,之后碰到相同的则快指针跳过,碰到不同的则标记新的tag。

class Solution {
    public int removeDuplicates(int[] nums) {  
        int fastIndex = 1;
        int slowIndex = 1;
        int count = 0;
        int tag = nums[1];  //题目为有序数组
        while(fastIndex < nums.length && slowIndex < nums.length - count){
            nums[slowIndex] = nums[fastIndex];
            if(nums[fastIndex] == tag){
                count++;
            }else{
                tag = nums[fastIndex];
                slowIndex++;
            }
            fastIndex++;
        }
        return nums.length - count;
    }
}

方法②:直接省去标记的问题,因为数组是有序的,慢指针和快指针所指向的元素只有nums[fastIndex] > nums[slowIndex]以及nums[fastIndex] == nums[slowIndex]两种情况,相等的时候快指针即可跳过,一旦不相等即可做赋值操作。

public int removeDuplicates(int[] nums) {
        //题目为有序数组
        int fastIndex = 0;
        int slowIndex = 0;
        while(fastIndex < nums.length){
            if(nums[fastIndex] > nums[slowIndex]){
                nums[++slowIndex] = nums[fastIndex];
            }else{
                fastIndex++;
            }
        }
        return slowIndex + 1;
    }

方法①与方法②速度差距:

在这里插入图片描述

[简单] 283. 移动零

原题链接

经典的双指针法。因为末尾要保留0,所以使用对换的swap方式来做。但这样时间效率不是最高的,直接覆盖,然后在末尾使用Arrays.fill(nums,count,nums.length,0);直接填充0能够更高效。

class Solution {
    public void moveZeroes(int[] nums) {
        int fastIndex = 0;
        int slowIndex = 0;
        while(fastIndex < nums.length){
            if(nums[fastIndex] != 0){
                int temp = nums[fastIndex];
                nums[fastIndex] = nums[slowIndex];
                nums[slowIndex] = temp;
                slowIndex++;
            }
            fastIndex++;
        }
    }
}

[简单] 844. 比较含退格的字符串

原题链接

方法①:也可以用快慢指针法,把两个字符串该删的删了对最后的结果作比较,这里就不重复实现了。

方法②:看到退格操作'#'就想到使用栈,用两个栈保存st经过退格操作之后的值,然后再出栈进行比较。

class Solution {
    public boolean backspaceCompare(String s, String t) {
        Stack<Character> sStack = new Stack<>();
        Stack<Character> tStack = new Stack<>();
        for(int i = 0; i < s.length(); i++){
            if(s.charAt(i) == '#' && !sStack.empty()){
                sStack.pop();
            }else if(s.charAt(i) != '#'){
                sStack.push(s.charAt(i));
            }
            //System.out.println(sStack.toString());
        }
        for(int i = 0; i < t.length(); i++){
            if(t.charAt(i) == '#' && !tStack.empty()){
                tStack.pop();
            }else if(t.charAt(i) != '#'){
                tStack.push(t.charAt(i));
            }
            //System.out.println(tStack.toString());
        }
        if(sStack.size() != tStack.size()) return false;
        while(!sStack.empty() && !tStack.empty()){
            if(sStack.pop() != tStack.pop()) return false;
        }
        return true;
    }
}

方法③:两个指针从后往前遍历,效率高,但是对边界的控制比较麻烦,没有栈写起来简单

class Solution {
    public boolean backspaceCompare(String s, String t) {
        int sIndex = s.length() - 1;
        int tIndex = t.length() - 1;
        int sCount = 0;
        int tCount = 0;
        while(sIndex >= 0 || tIndex >= 0){
            //让sIndex指向s中接下来要比较的字符(能够确定最后不被删除
            while (sIndex >= 0){
                if (s.charAt(sIndex) == '#') {
                    sCount++;
                    sIndex--;
                } else if (sCount > 0) {
                    sCount--;
                    sIndex--;
                } else{
                    break;
                }
            }
            while (tIndex >= 0){
                if (t.charAt(tIndex) == '#') {
                    tCount++;
                    tIndex--;
                } else if (tCount > 0) {
                    tCount--;
                    tIndex--;
                } else{
                    break;
                }
            }
            if(sIndex < 0 || tIndex < 0) break;
            if (s.charAt(sIndex) != t.charAt(tIndex)){
                return false;
            }
            sIndex--;
            tIndex--;
        }
        if(sIndex >=0 || tIndex >= 0) return false;
        return true;
    }
}

[简单] 977. 有序数组的平方

原题链接

题目中没有强调在原数组上修改,返回值也是int[],就可以思考是不是创建一个新的数组更方便一些。
找到数组中正负值的分界点,然后从分界点向两边遍历,按照绝对值的大小挨个平方计算后加入到新的数组中。正数或者负数用完之后无法继续进行绝对值比较的循环。然后再将左边或者右边剩下的数依次放入答案数组中。
时间复杂度O(n)

class Solution {
    public int[] sortedSquares(int[] nums) {
        int[] answer = new int[nums.length];
        int zeroIndex = 0;
        //找到正负分界点
        while(zeroIndex < nums.length && nums[zeroIndex] < 0){
            zeroIndex++;
        }
        int negativeIndex = zeroIndex - 1;
        int positiveIndex = zeroIndex;
        int i = 0;
        while(negativeIndex >= 0 && positiveIndex < nums.length){
            if (Math.abs(nums[negativeIndex]) < Math.abs(nums[positiveIndex])) {
                answer[i++] = nums[negativeIndex] * nums[negativeIndex];
                negativeIndex--;
                continue;
            } else {
                answer[i++] = nums[positiveIndex] * nums[positiveIndex];
                positiveIndex++;
                continue;
            }
        }
        while(negativeIndex >= 0){
            answer[i++] = nums[negativeIndex] * nums[negativeIndex];
            negativeIndex--;
        }
        while(positiveIndex < nums.length){
            answer[i++] = nums[positiveIndex] * nums[positiveIndex];
            positiveIndex++;
        }
        return answer;
    }
}

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

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

相关文章

红黑树——插入底层实现【C++】面试重灾区!!

目录 前言 一&#xff0c;概念 定义 二&#xff0c;insert 情况一&#xff1a; 情况二&#xff1a; 情况三&#xff1a; insert代码 三&#xff0c; 红黑树验证(面试题) 产生随机数验证 每日一图区&#xff1a; 前言 AVL树是一棵绝对平衡的二叉搜索树&#xff0c;其…

IM 系统通信系统是什么

IM&#xff08;Instant Messaging&#xff09;系统是一种实时通信系统&#xff0c;允许用户通过互联网或内部网络即时发送文本消息、文件、音频、视频和其他类型的数据给其他用户或群组。这类系统通常支持在线状态显示、消息确认、消息历史记录和多端同步等功能&#xff0c;以提…

WEB3 在 React搭建的Dapp中通过redux全局获取并存储用户ETH与自定义token与交易所存储数量

上文 web3 在React dapp中全局管理web3当前登录用户/智能合约等信息中 我们简单操作&#xff0c;将web3的公共信息都存在了window对象上 然后 我们先来启动一下环境 终端输入 ganache -d打开项目 终端输入 truffle migrate --reset在区块链上发布一下智能合约 然后 我们在…

FedAT:分层机制更新的联邦学习

文章链接&#xff1a;FedAT: A Communication-Efficient Federated Learning Method with Asynchronous Tiers under Non-IID Data 发表会议: SC’21 (International Conference for High Performance Computing, Networking, Storage, and Analysis) 高性能计算&#xff0c;体…

Redis安装与常用命令

目录 一、Reids简介 二、Redis安装 2.1 Linux安装 2.2 Windows安装 三、Redis常用命令 3.1 Redis字符串 3.2 Redis哈希(Hash) 3.3 Redis列表&#xff08;List&#xff09; 3.4 Redis集合&#xff08;Set&#xff09; 一、Reids简介 Redis&#xff08;Remote Dictiona…

11.Z-Stack协议栈使用

f8wConfig.cfg文件 选择信道、设置PAN ID 选择信道 #define DEFAULT_CHANLIST 0x00000800 DEFAULT_CHANLIST 表明Zigbee模块要工作的网络&#xff0c;当有多个信道参数值进行或操作之后&#xff0c;把结果作为 DEFAULT_CHANLIST值 对于路由器、终端、协调器的意义&#xff1…

【Linux网络】Linux网络抓包工具tcpdump

一、tcpdump介绍 tcpdump 是一个Linux的网络抓包工具。它允许用户拦截和显示发送或收到过网络连接到该计算机的TCP/IP和其他数据包。tcpdump 适用于大多数的类Unix系统操作系统(如linux,BSD等)。 二、安装&用法说明 1&#xff09;安装 $ yum -y install tcpdump2&#…

【知网检索征稿】第九届社会科学与经济发展国际学术会议 (ICSSED 2024)

第九届社会科学与经济发展国际学术会议 (ICSSED 2024) 2024 9th International Conference on Social Sciences and Economic Development 第九届社会科学与经济发展国际学术会议(ICSSED 2024)定于2024年3月22-24日在中国北京隆重举行。会议主要围绕社会科学与经济发展等研究…

Everything结合内网穿透搭建在线资料库,一秒实现随时随地访问

Everythingcpolar搭建在线资料库&#xff0c;实现随时随地访问 文章目录 Everythingcpolar搭建在线资料库&#xff0c;实现随时随地访问前言1.软件安装完成后&#xff0c;打开Everything2.登录cpolar官网 设置空白数据隧道3.将空白数据隧道与本地Everything软件结合起来总结 前…

(免费领源码)Java#MYSQL在线学习平台09650-计算机毕业设计项目选题推荐

目 录 摘要 1 绪论 1.1 选题背景及意义 1.2国内外现状分析 1.3论文结构与章节安排 2 在线学习平台系统分析 2.1 可行性分析 2.2 系统业务流程分析 2.3 系统功能分析 2.3.1 功能性分析 2.3.2 非功能性分析 2.4 系统用例分析 2.5本章小结 3 在线学习平台总体设计 …

nginx 常用优化配置项教程

本文目录 跨域配置 动静分离 反向代理-负载均衡 配置SLL证书 资源压缩 缓存机制 IP黑白名单 防盗链 大文件传输优化 跨域问题 产生原因 产生跨域问题的主要原因就在于 「同源策略」 &#xff0c;为了保证用户信息安全&#xff0c;防止恶意网站窃取数据&#xff0c;…

每日一题 2103. 环和杆 (简单)

简单题&#xff0c;直接统计就行 class Solution:def countPoints(self, rings: str) -> int:n len(rings) // 2cnt [[0, 0, 0] for _ in range(10)]for i in range(0, 2*n, 2):if rings[i] "R":cnt[int(rings[i 1])][0] 1if rings[i] "G":cnt[i…

阿里云多款ECS产品全面升级 性能最多提升40%

“阿里云始终围绕‘稳定、安全、性能、成本、弹性’的目标不断创新&#xff0c;为客户创造业务价值。”10月31日&#xff0c;杭州云栖大会上&#xff0c;阿里云弹性计算计算产品线负责人张献涛表示&#xff0c;通过持续的产品和技术创新&#xff0c;阿里云发布了HPC优化实例等多…

conda命令克隆(复制)环境方法及问题解决

背景&#xff1a;与同事进行工作交接&#xff0c;在服务器上想直接拷贝他的环境过来&#xff0c;直接复制anaconda3/envs里的环境包过来后续安装包的时候由于路径是原来的路径会安装到对方环境里&#xff0c;因此使用conda命令来进行环境克隆。 &#xff08;Linux/服务器中&am…

【广州华锐互动】有机化合物结构3D虚拟展示帮助学生更好地理解和掌握复杂化学知识

随着科技的发展&#xff0c;我们的学习方式正在发生深刻的变化。其中&#xff0c;有机化合物结构3D虚拟展示作为一种新兴的教学方式&#xff0c;正在逐渐被广泛接受和应用。这种新型的展示方式对于提高学生的学习效率&#xff0c;增强学习体验&#xff0c;以及推动化学教育的发…

十六章反射与注解总结

16.1 反射 反射&#xff08;Reflection&#xff09;是指在运行时获取类的信息&#xff0c;并可以动态调用类的方法、访问或修改类的属性&#xff0c;以及构造对象的能力。 Java的反射提供了一套API&#xff0c;允许你在运行时检查类的结构、调用类的方法、获取和设置类的属性&…

curl(一)基础

一 基础入门 ① 本文讲解curl的版本 curl支持的协议 国密curl ② 升级curl 后续&#xff1a; 注意新版本的新特性备注&#xff1a; 升级的必要性 yum 升级 curl 源码编译安装curl ③ curl排错参数 1、-v 或 --verbose --> 重点 2、--trace 和 --trace-ascii --…

小红书薯多功能引流脚本,精准引流工具,内置私信点赞关注评论回复各项功能【引流软件+引流教程】

软件介绍&#xff1a; 多个高级功能精准引流 精准留痕引流&#xff0c;功能非常多 设备需求&#xff1a; 适用于安卓7.0版本&#xff0c;鸿蒙&#xff0c;模拟器均可使用 文章分享者&#xff1a;Linxiaoyu2022 文章的来源&#xff1a;vipwz.blog.csdn.net 有任何互联网问…

Java作业二

一、使用方法编写求圆面积和周长的程序&#xff0c;运行时提示输入圆半径&#xff0c;然后输出计算结果。运行效果如下图所示&#xff1a; import java.util.Scanner;public class Test {public static void main(String[] args) {Scanner input new Scanner(System.in);Syste…

安卓抓包之小黄鸟

下载安装 下载地址: https://download.csdn.net/download/yijianxiangde100/88496463 安装apk 即可。 证书配置: