2024-03-13 代码随想录

代码随想录算法训练营day8 | 题目344、题目541、卡码网54、题目151、卡码网55


题目一描述

344. 反转字符串

编写一个函数,其作用是将输入的字符串反转过来。输入字符串以字符数组 s 的形式给出。
不要给另外的数组分配额外的空间,你必须原地修改输入数组、使用 O(1) 的额外空间解决这一问题。

示例 1:

输入:s = ["h","e","l","l","o"]
输出:["o","l","l","e","h"]
示例 2:

输入:s = ["H","a","n","n","a","h"]
输出:["h","a","n","n","a","H"]

提示:

1 <= s.length <= 10^5
s[i] 都是 ASCII 码表中的可打印字符

解题思路

双指针在数组的运用,两边相向移动。

代码实现

方法一:

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

题目二描述

541. 反转字符串 II

给定一个字符串 s 和一个整数 k,从字符串开头算起,每计数至 2k 个字符,就反转这 2k 字符中的前 k 个字符。

如果剩余字符少于 k 个,则将剩余字符全部反转。
如果剩余字符小于 2k 但大于或等于 k 个,则反转前 k 个字符,其余字符保持原样。

示例 1:
输入:s = "abcdefg", k = 2
输出:"bacdfeg"
示例 2:
输入:s = "abcd", k = 2
输出:"bacd"

提示:
1 <= s.length <= 104
s 仅由小写英文组成
1 <= k <= 104

解题思路

字符串转为字符数组是常用操作
把字符串划分为区间,反转每个区间的前几个字母。

代码实现

方法一:

class Solution {
    public String reverseStr(String s, int k) {
        char[] ss = s.toCharArray();
        for (int i = 0; i < ss.length; i += 2 * k) {
            int left = i;
            int right = Math.min(i + k - 1, ss.length - 1);
            while (left < right) {
                char temp = ss[left];
                ss[left] = ss[right];
                ss[right] = temp;
                left++;
                right--;
            }
        }
        String res = new String(ss);
        return res;
    }
}

题目三描述

54. 替换数字

时间限制:1.000S 空间限制:128MB
题目描述
给定一个字符串 s,它包含小写字母和数字字符,请编写一个函数,将字符串中的字母字符保持不变,而将每个数字字符替换为number。 例如,对于输入字符串 "a1b2c3",函数应该将其转换为 "anumberbnumbercnumber"。
输入描述
输入一个字符串 s,s 仅包含小写字母和数字字符。
输出描述
打印一个新的字符串,其中每个数字字符都被替换为了number
输入示例
a1b2c3
输出示例
anumberbnumbercnumber
提示信息
数据范围:
1 <= s.length < 10000。

解题思路

遍历然后判断数据类型,重点在于如何判断以及不定长字符串的构建。

代码实现

方法一:

import java.util.*;

class Main {
    public static void main (String[] args) {
        Scanner sc = new Scanner(System.in);
        String s = sc.nextLine();
        char[] ss = s.toCharArray();
        StringBuilder sb = new StringBuilder();
        for(char c: ss){ // 应该用char而不是Char
            if(Character.isDigit(c)){
                sb.append("number");
            }else if(Character.isLetter(c)){
                sb.append(c);
            }
        }
        // return sb.toString(); // main 方法不能返回值
        System.out.println(sb.toString());
    }
}

技巧总结

手写算法,学会import java.util.*;
学会Scaner
学会StringBuilder,其转为字符串要用toString(),而char[] 要用String.valueOf 或者 new String
注意遍历是小写,定义map或者set,list才是Character
学会判断数据类型,Character.isDigitCharacter.isLetter


题目四描述

151. 反转字符串中的单词

给你一个字符串 s ,请你反转字符串中 单词 的顺序。

单词 是由非空格字符组成的字符串。s 中使用至少一个空格将字符串中的 单词 分隔开。

返回 单词 顺序颠倒且 单词 之间用单个空格连接的结果字符串。

注意:输入字符串 s中可能会存在前导空格、尾随空格或者单词间的多个空格。返回的结果字符串中,单词间应当仅用单个空格分隔,且不包含任何额外的空格。

示例 1:
输入:s = "the sky is blue"
输出:"blue is sky the"
示例 2:
输入:s = " hello world "
输出:"world hello"
解释:反转后的字符串中不能存在前导空格和尾随空格。
示例 3:
输入:s = "a good example"
输出:"example good a"
解释:如果两个单词间有多余的空格,反转后的字符串需要将单词间的空格减少到仅有一个。

提示:
1 <= s.length <= 10^4
s 包含英文大小写字母、数字和空格 ' '
s 中 至少存在一个 单词

进阶:如果字符串在你使用的编程语言中是一种可变数据类型,请尝试使用 O(1) 额外空间复杂度的 原地 解法。

解题思路

直接的想法是扫描出所有的单词,压入栈内,再取出后在中间填加空格。
第二个思路是先把整体所有字符串反转,再找到每个单词对其进行反转,然后去掉首尾的和中间的多余的空格。

代码实现

方法一:

class Solution {
    public String reverseWords(String s) {
        StringBuilder sb = new StringBuilder();
        Stack<String> stack = new Stack<>();
        char[] ss = s.toCharArray();
        int start = 0;
        while (start < ss.length) {
            while (start < ss.length && ss[start] == ' ') {
                start++;
            }
            StringBuilder word = new StringBuilder();
            while (start < ss.length && ss[start] != ' ') {
                word.append(ss[start]);
                start++;
            }
            if (word.length() != 0)
                stack.push(word.toString());
        }
        while (!stack.empty()) {
            sb.append(stack.pop());
            if (!stack.empty())
                sb.append(' ');
        }
        return sb.toString();
    }
}

方法二:

class Solution {
    public String reverseWords(String s) {
        char[] ss = s.toCharArray();
        reverse(ss, 0, ss.length - 1);
        // ss = reverse(ss, 0, ss.length - 1); // 没必要接收返回值
        int start = 0;
        int end = 0;
        while (start < ss.length) {
            while (start < ss.length && ss[start] == ' ') {
                start++;
            }
            end = start;
            while (end < ss.length && ss[end] != ' ') {
                end++;
            }
            reverse(ss, start, end - 1);
            start = end;
        }
        // 去空格
        StringBuilder sb = new StringBuilder();
        boolean flag = false;
        int left = 0;
        int right = ss.length - 1;
        while (ss[left] == ' ') {
            left++;
        }
        while (ss[right] == ' ') {
            right--;
        }
        for (int i = left; i < right + 1; i++) {
            char c = ss[i];
            if (c != ' ' || c == ' ' && ss[i - 1] != ' ') {
                sb.append(c);
            }
        }
        return String.valueOf(sb);
    }

    public void reverse(char[] ss, int start, int end) {
        while (start < end) {
            char temp = ss[start];
            ss[start] = ss[end];
            ss[end] = temp;
            start++;
            end--;
        }
    }

}

技巧总结

栈的初始化,进出栈与判空
引用类型被当作形参传递,直接就会被修改。


题目五描述

55. 右旋字符串

时间限制:1.000S 空间限制:128MB
题目描述
字符串的右旋转操作是把字符串尾部的若干个字符转移到字符串的前面。给定一个字符串 s 和一个正整数 k,请编写一个函数,将字符串中的后面 k 个字符移到字符串的前面,实现字符串的右旋转操作。

例如,对于输入字符串 "abcdefg" 和整数 2,函数应该将其转换为 "fgabcde"。

输入描述
输入共包含两行,第一行为一个正整数 k,代表右旋转的位数。第二行为字符串 s,代表需要旋转的字符串。

输出描述
输出共一行,为进行了右旋转操作后的字符串。

输入示例
2
abcdefg

输出示例
fgabcde

提示信息
数据范围:
1 <= k < 10000,
1 <= s.length < 10000;

解题思路

先反转前一部分,再反转后一部分,再反转全部

代码实现

方法一:

import java.util.*;

class Main{
    public static void main(String[] args){
        Scanner sc = new Scanner(System.in);
        int k = Integer.valueOf(sc.nextLine());
        String s = sc.nextLine();
        char[] ss = s.toCharArray();
        
        int start = 0;
        int end = ss.length - k - 1;
        reverse(ss, start, end);
        
        start = end + 1;
        end = ss.length - 1;
        reverse(ss, start, end);
        
        reverse(ss, 0, ss.length - 1);
        
        System.out.println(ss);
    }
    
  
    public static void reverse(char[] ss, int start, int end){
        while(start < end){
            char temp = ss[start];
            ss[start] = ss[end];
            ss[end] = temp;
            start++;
            end--;
        }
    }
}

技巧总结

判断下标减几的时候,最好还是用例子试试。


最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,233评论 6 495
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,357评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,831评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,313评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,417评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,470评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,482评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,265评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,708评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,997评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,176评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,827评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,503评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,150评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,391评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,034评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,063评论 2 352

推荐阅读更多精彩内容