代码随想录算法训练营Day6 | 344.反转字符串、541.反转字符串||、替换数字、151.反转字符串中的单词、右旋字符串

news2024/12/24 9:02:02

LeetCode 344 反转字符串

在这里插入图片描述
本题思路:反转字符串比较简单,定义两个指针,一个 i = 0, 一个 j = s.length-1。然后定义一个临时变量 tmp,进行交换 s[i] 和 s[j]。

class Solution {
    public void reverseString(char[] s) {
        int i = 0;
        int j = s.length-1;
        while(i <= j){
            char tmp = s[i];
            s[i] = s[j];
            s[j] = tmp;
            i++;
            j--;
        }
    }
}

LeetCode 541 反转字符串||

在这里插入图片描述

本题思路:最重要的理解题目含义,就是每一次走 2*k 步,如果剩余的字符长度大于等于k,就反转前 k 个字符,如果小于 k , 就将剩下的全部反转。 个人感觉这个剩余字符小于 2k的条件毫无软用,

下面用一个例子来分析下思路,方便更好理解。

  • 初始情况下 i = 0, i 每次走 i += 2 * k 步在这里插入图片描述
  • 判断剩余个数是否大于等于 k =2,其实就是判断(i + k <= array.length) 此时符合条件,反转前 K 个 ,调用 reverseSubStr(i,i+k,array),左闭右开。 在这里插入图片描述
  • i 往后移动 i = i + 2 * k,此时 i = 0 + 4 = 4在这里插入图片描述
  • 继续判断 (i + k <= array.length) ,此时条件符合 4 + 2 <= 6, 反转前 K 个 ,调用 reverseSubStr(i,i+k,array),左闭右开。 在这里插入图片描述
  • 此时 i = i + 2 * k = 4 + 8,不符合循环,结束。最终结果如下在这里插入图片描述
class Solution {
    public String reverseStr(String s, int k) {
        // 将字符串转换成数组
        char[] array = s.toCharArray();

        for(int i = 0; i < array.length; i+=2*k){
            // 先判断如果剩余字符个数大于 k,或者小于 2k, 就反转这前 K个
            // 本题主要理清楚,只要剩余个数大于等于k,
            if(i + k <= array.length){
                reverseSubStr(i,i+k,array);
                continue;
            }
            // 如果小于 K,就反转剩下的全部
            reverseSubStr(i,array.length,array);
        }
        return new String(array);
    }

    // 左闭右开
    public static void reverseSubStr(int l,int r, char[] s){
        while(l <= r-1){
            char tmp = s[l];
            s[l] = s[r-1];
            s[r-1] = tmp;
            l++;
            r--;
        }
    }
}

KamaCoder 替换数字

在这里插入图片描述
本题思路:使用字符串拼接,使用 StringBuilder,遍历字符串,如果是 小写字母的话直接拼接,即可,如果是数字的话,就拼接 “number”

public class Main{
    public static void main(String[] args){
        Scanner scanner = new Scanner(System.in);
        String s = scanner.nextLine();
        StringBuilder sb = new StringBuilder();
        for( Character c  : s.toCharArray()){
            if(c >= 'a' && c <= 'z' ){
                sb.append(c);
            }else{
                sb.append("number");
            }
        }
        System.out.println(sb);
    }
}

LeetCode 151 反转字符串中的单词

在这里插入图片描述
本题思路

  1. 去掉前面、后面、单词之间的重复空格
  2. 先整体反转
  3. 再单个单词反转

通过示例 2 ,来画个图分析下思路,下面是初始状态在这里插入图片描述

  • 首先是去空格判断,定义一个 Trim() 方法,来进行去除两端的空格,通过定义两个指针,一个 left ,一个 right, left++, right–。碰到第一个不为空格的就停下。在这里插入图片描述
  • 这是去掉两端空格以后的字符串 s在这里插入图片描述
  • 接下里就要进行字符串之间的多余空格进行去重,遍历字符串 s 分为两步; 新定义一个字符串拼接 StringBuilder sb
    • 如果不是空格的话,就直接拼接 sb.append(s.charAt(left)),如下,left++,直到遇到空格停止在这里插入图片描述
    • 可以看到上图,此时就遇到了空格,因为可能有多个空格,我们需要保留一个空格,所以需要判断 (left-1) 这个元素是不是空格,如果是空格,说明空格已经保存了,就跳过不做任何处理,直到遇到的不是空格为止, 可以看到上图 left-1 元素不是空格,所以直接拼接空格,并且left++;在这里插入图片描述
    • 此时上图,又遇到不是空格,直接拼接在这里插入图片描述
  • 最终得到 sb, 去掉多余空格以后的新字符串 sb在这里插入图片描述

  • 接下来就是反转全部字符串,这个逻辑的话和上述反转字符串一样,定义两个指针,一个 left, 一个 right, 然后定义一个 临时变量来交换 这两个数字
  • 最后反转每一个单词。

此处,使用 StringBuilder的时候, 不能 sb.charAt(left) = sb.charAt(right),会有语法错误。必须使用 sb.setCharAt(原值,要改变的值)

class Solution {
    public String reverseWords(String s) {
        // 1. 先去掉多余的空格
        StringBuilder sb = Trim(s);
        // 2. 先反转全部
        Reverse(sb,0,sb.length()-1);
        // 3. 反转每一个单词;

        // 定义反转起始位置
        int start = 0;
        // 定义反转结束位置
        int end = 0;
        for (int i = 0; i < sb.length(); i++) {
            // 找到第一个为空格的下标
            if (sb.charAt(i) == ' ') {
                end = i - 1;
                Reverse(sb, start, end);
                start = end + 2;
            }
        }
        // 最后一个单词单独进行反转
        Reverse(sb,start,sb.length()-1);
        return sb.toString();
    }
    // 反转所有单词
    public static void Reverse(StringBuilder s, int start, int end) {
        int left = start;
        int right = end;
        while (left <= right) {
            char tmp = s.charAt(left);
            s.setCharAt(left, s.charAt(right));
            s.setCharAt(right, tmp);
            left++;
            right--;
        }
    }

    // 去空格
    public static StringBuilder Trim(String s) {
        int left = 0;
        int right = s.length() - 1;
        StringBuilder sb = new StringBuilder();
        // 去除前面的空格
        while (left <= right && s.charAt(left) == ' ') {
            left++;
        }
        // 去除后面的空格
        while (left <= right && s.charAt(right) == ' ') {
            right--;
        }
        // 去重字符串之间的多余空格
        while (left <= right) {
            // 如果不等于空格,直接拼接这个字符
            if (s.charAt(left) != ' ') {
                sb.append(s.charAt(left));
                left++;
            } else { // 此时等于空格,需要判断前面一个是否为空格
                if (s.charAt(left - 1) != ' ') {
                    sb.append(s.charAt(left));
                    left++;
                } else {
                    left++;
                }
            }
        }
        return sb;
    }
}

虽然整体逻辑不难,但是写代码的时候,出现了很多错误,考虑不周到。还得多写几遍,没那么简单。

KamaCoder 右旋字符串

在这里插入图片描述
本题思路

  1. 先整体反转
  2. 再反转前 k 个
  3. 再反转后 length - k 个

可以通过下图分析得到,为什么这么做!在这里插入图片描述

public static void main (String[] args) {
        /* code */
        Scanner scanner = new Scanner(System.in);
        String s = scanner.nextLine();
        int k = scanner.nextInt();     
        // 1. 先整体旋转
        s = reverse(s,0,s.length()-1);
        // 2. 再分别反转
        s = reverse(s,0,k-1);
        s = reverse(s,k,s.length()-1);
        System.out.println(s);

    }

    // 此处左闭右闭
    public static String reverse(String s, int start,int end){
        char[] array = s.toCharArray();
        int left = start;
        int right = end;
        while (left < right) {
            char tmp = array[left];
            array[left] = array[right];
            array[right] = tmp;

            left++;
            right--;
        }
        return new String(array);
    }

写这道题的时候,直接用 String 来进行交换,出现了语法错误。因为 String 是不可变的,不能交换里面的内容!

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

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

相关文章

工程项目管理系统源码:Java版工程项目管理系统平台助力工程企业迈向数字化管理的巅峰

随着企业规模的不断扩大和业务的快速发展&#xff0c;传统的工程项目管理方式已经无法满足现代企业的需求。为了提高工程管理效率、减轻劳动强度、提高信息处理速度和准确性&#xff0c;企业需要借助先进的数字化技术进行转型。本文将介绍一款采用Spring CloudSpring BootMybat…

创新铸就卓越 HashData再度荣获 IT168技术卓越奖

12月20日&#xff0c;由国内专业数据库及架构技术社区 ITPUB、 IT 垂直门户IT168 联合主办的“2023 技术卓越奖”评选结果正式揭晓&#xff0c; 酷克数据基于云原生架构打造的HashData云数仓&#xff0c;凭借创新的技术架构、强大的性能表现、行业领先的落地规模&#xff0c;连…

在GitHub找开源项目

在 GitHub 的搜索框里&#xff1a; 使用搜索关键词可以在 GitHub 上快速的找你需要的开源项目&#xff1a; 限制搜索范围 通过 in 关键词 (大小写不敏感) 限制搜索范围&#xff1a; 公式搜索范围in:name xxx项目名包含xxxin:description xxx项目描述包含xxxin:readme xxx项目…

记录 | gdb调试的基本命令

r (run) 运行程序 b (breakpoint) 打断点&#xff0c;比如 b func(打到函数) b 5(打到第5行)(当前文件) b main.cpp:5(main.cpp的第5行) b MyClass::func() (打到类的成员函数func()、在类内可以 p this 打印、p *this、p this->name) tb (temporary breakpoint) 临时断…

算法基础之欧拉函数

欧拉函数 核心思想&#xff1a;欧拉函数: 证明 &#xff1a;容斥原理 #include<iostream>#include<algorithm>using namespace std;const int N 110;int main(){int n;cin>>n;while(n--){int a;cin>>a;int res a;for(int i 2;i < a/i; i){if(a %…

使用TikTok云手机轻松拓展全球市场

TikTok作为一款风靡全球的短视频应用&#xff0c;全球影响力不断扩大。越来越多的商家开始借助TikTok分享作品、在海外市场上获取商业机会。要想更好地借助TikTok扩大海外市场&#xff0c;使用TikTok云手机是一个好选择。本文将介绍TikTok云手机的几大作用&#xff0c;以助您更…

解决ESP8266无法退出透传问题以及获取网络时间以及天气方法

网上很多配置ESP8266的教程&#xff0c;但是遇到无法退出透传模式的情况却没有找得到答案&#xff0c;不知道是大家都没遇到还是怎么样&#xff0c;以下是我的解决方法&#xff1a;实测有效 先发送“”&#xff08;三个加号&#xff09;&#xff08;如果是在串口调试助手调试&…

保存Google Colab数据文件或目录到Google Drive云盘

一、背景 在Google Colab上使用T4 GPU完成微调训练的模型&#xff0c;保存了模型和训练状态的文件存放在Google Colab /content目录内&#xff0c;需要备份。考虑到下载到本地需要大量的流量&#xff0c;且下次继续训练还会上传&#xff0c;最后确定直接保存到Google Drive&am…

linux 上安装 minio

第一步&#xff0c;下载 wget https://dl.minio.org.cn/server/minio/release/linux-amd64/minio 第二步&#xff0c;修改权限 chmod x minio 第三步&#xff0c;设置 Path mv minio /usr/local/bin/ 第四步&#xff0c;创建 minio mkdir minio 第五步&#xff0c;启动 …

C#中var、object和dynamic的区别

在C#编程语言中&#xff0c;我们经常会遇到var、object和dynamic这三个关键字。它们都用于声明变量&#xff0c;但在使用方法和特性上存在一些重要的区别。本文将详细介绍这三者的差异。 目录 var关键字object关键字dynamic关键字总结 var关键字 var是C#语言中的隐式类型推断…

JDBC学习,从入门到入土

JDBC引入 JDBC概念&#xff1a; JDBC是使用Java语言操作关系型数据库的一套API。全称&#xff1a;&#xff08;Java DataBase Connectivity&#xff09;Java数据库连接 JDBC的本质&#xff1a; 官方定义的一套操作所有关系型数据库的规则&#xff0c;即接口。 各个数据库厂…

Windows操作系统重装【老毛桃、软碟通、硬盘装机以及装机原理介绍】

目录 一、老毛桃装机 1.1 老毛桃介绍 1.2 注意事项 1.3 操作步骤 1.3.1 老毛桃U盘制作 1.3.2 老毛桃u盘启动 1.3.3 老毛桃u盘重装系统 二、软碟通装机 2.1 软碟通介绍 2.2 主要特点及功能 2.3 操作步骤 2.3.1 用软碟通制作U盘安装盘 2.3.2 U盘启动 2.3.3 安装系统…

构建陪诊预约系统:技术实现与用户体验

在医疗服务不断创新的背景下&#xff0c;陪诊预约系统作为一种结合技术与人性化服务的应用&#xff0c;为患者提供了更为便捷和贴心的医疗体验。让我们通过简单的示例代码&#xff0c;了解一下如何构建一个基本的陪诊预约系统。 技术栈选择 在开始构建陪诊预约系统之前&…

【数据结构和算法】子数组最大平均数 I

其他系列文章导航 Java基础合集数据结构与算法合集 设计模式合集 多线程合集 分布式合集 ES合集 文章目录 其他系列文章导航 文章目录 前言 一、题目描述 二、题解 2.1 滑动窗口含义 2.2 滑动窗口一般解法 2.3 方法一&#xff1a;滑动窗口 三、代码 3.1 方法一&#…

qt-C++笔记之app.processEvents()和QApplication::processEvents()的区别

qt-C笔记之app.processEvents()和QApplication::processEvents()的区别 code review! 代码1&#xff1a; QApplication app(argc, argv); app.processEvents(); 代码2: QApplication::processEvents(); 区别 代码1和代码2的区别在于代码1中使用了一个具体的QApplication对…

26--字符流与字节流

1、IO流概述 1.1 什么是IO流 Java中I/O操作主要是指使用java.io包下的内容&#xff0c;进行输入、输出操作。输入也叫做读取数据&#xff0c;输出也叫做作写出数据。我们把这种数据的传输&#xff0c;可以看做是一种数据的流动&#xff0c;按照流动的方向&#xff0c;以内存为…

力扣刷题记录(17)LeetCode:416、1049

416. 分割等和子集 可以将该问题看成是一个背包问题。背包的容量就是nums数组和的一半。我们如果能够将背包装满就意味着可以将数组分割成两个元素和相等的子集。 1.确定dp[i]的含义 索引i表示背包的容量&#xff0c;dp[i]表示当前容量能够装载的最大值 2.确定动态转移方程 …

JS - 设计模式持续学习中

通过例子持续学习JS设计模式中&#xff0c;接下来请跟随我的步伐走进我的学习笔记世界~ 什么是设计模式&#xff1f;我们为什么需要学习设计模式&#xff1f; 设计模式是可以更好解决问题的一种方案。 这意味着什么&#xff1f;如果你开发的项目的功能是固定的&#xff0c;永…

算法通关村第十四关—堆结构(青铜)

堆结构 一、堆的概念和特征 堆是将一组数据按照完全二叉树的存储顺序&#xff0c;将数据存储在一个一维数组中的结构。堆有两种结构&#xff0c;一种称为大顶堆&#xff0c;一种称为小顶堆&#xff0c;如下图。 1.小顶堆&#xff1a;任意节点的值均小于等于它的左右孩子&#…

android 新版studio gradle 里面没有offline 勾选项

studio 右边 gradle 上面有个图标可以点击切换