代码随想录:字符串1-4

344.反转字符串

题目

        编写一个函数,其作用是将输入的字符串反转过来。输入字符串以字符数组 char[] 的形式给出。不要给另外的数组分配额外的空间,你必须原地修改输入数组、使用 O(1) 的额外空间解决这一问题。你可以假设数组中的所有字符都是 ASCII 码表中的可打印字符。

        示例 1:
        输入:["h","e","l","l","o"]
        输出:["o","l","l","e","h"]

代码

class Solution {
    public void reverseString(char[] s) {
        int left = 0;  //指向数组头部
        int right = s.length - 1;  //指向数组尾部
        while(left < right){
            char tmp = s[left];  //交换元素
            s[left] = s[right];
            s[right] =tmp;
            left++;  //指针向中间移动
            right--;
        }
    }
}

总结

1.原理

        题目要求是把char数组进行翻转,直观理解就是把头尾元素交换,一直向中间移动,可以用双指针实现。

2.算法流程

        用left指向第一个元素,right指向数组最后一个元素,交换left和right指向的元素,然后left++右移,right--左移,直到不满足left<right时跳出循环。

3.注意点

        循环结束的条件是left<right,不需要写等号,因为头尾交换元素的时候,其下标是不需要重合的。

第二次刷题总结

1.一遍过啦!char数组获取长度ch.length即可,没有括号()。

541.反转字符串II

题目

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

        如果剩余字符少于 k 个,则将剩余字符全部反转。

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

代码(StringBuilder)

class Solution {
    public String reverseStr(String s, int k) {
       StringBuilder sb = new StringBuilder(s);
        for(int i = 0;i < s.length();i += 2 * k){  //i每次移动2k
            if(i + k <= s.length()){  //如果当前长度有k
                reverse(sb,i,i+k-1);  //反转k个
            }
            else{  //如果当前长度步骤k
                reverse(sb,i,s.length()-1);  //反转当前元素到末尾元素
            }
        }
        return sb.toString();
    }
    public void reverse(StringBuilder sb,int start,int end){
        while(start < end){
            char tmp = sb.charAt(start);
            sb.setCharAt(start,sb.charAt(end));
            sb.setCharAt(end,tmp);
            start++;
            end--;
        }
    }
}

代码(char数组)

class Solution {
    public String reverseStr(String s, int k) {
        char[] chs = s.toCharArray();
        for(int i=0; i < chs.length; i += 2*k){
            if(i+k <= chs.length){
                reverse(chs,i,i+k-1);
            }
            else{
                reverse(chs,i,chs.length-1);
            }
        }
        return new String(chs);
    }
    
    public void reverse(char[] chs, int i, int j){
        while(i < j){
            char tmp = chs[i];
            chs[i] = chs[j];
            chs[j] = tmp;
            i++;
            j--;
        }
    }
}

总结

1.原理

        题目要求索引i字符串的开始,每次移动2k,并翻转着2k个元素的前k个。核心是索引以2k为步长进行移动,且翻转字符时要分两种情况进行判断。两种情况如下,如果当前2k前面有k个元素,就直接翻转当前索引i后边的k个元素。但如果当前2k的前面不足k的元素,就只能翻转当前索引i到最后一个元素。

2.算法流程

        for循环遍历s字符串,用i=0作为初始索引,每次增加2k直到跳出循环。进入循环后,会根据当前能否满足k个字符的翻转进行分支。首先判断i+k是不是满足长度能够翻转k个元素,即判断i + k <= s.length(),如果满足,就调用函数reverse(sb,i,i+k-1),翻转k个字符。如果不满足,调用函数reverse(sb,i,s.length()-1),翻转当前元素到末尾元素。

        reverse函数就是题目344的原理,用头尾指针一边交换一边向中间移动即可。

3.注意点

        由于Java的String不可修改,所以最开始将String转为Stringbuild对象,然后进行翻转操作,最后用toString()返回String类型即可。

4.语法问题

        StringBuild对象的元素修改要用setCharAt()函数。

第二次刷题总结

1.算法没有问题,但是没有使用StringBuilder对象,直接在String上操作,结果编译出错了。要记住,String是不能修改的,而reverse需要用setCharAt()函数修改字符串的内容,因此必须用StringBuilder。

替换数字

题目

        给定一个字符串 s,它包含小写字母和数字字符,请编写一个函数,将字符串中的字母字符保持不变,而将每个数字字符替换为number。

        例如,对于输入字符串 "a1b2c3",函数应该将其转换为 "anumberbnumbercnumber"。

        对于输入字符串 "a5b",函数应该将其转换为 "anumberb"

代码

import java.util.Scanner;

class Main{
    public static void main(String[] args){
        Scanner sc = new Scanner(System.in);
        String s = sc.next();
        StringBuilder sb = new StringBuilder(); //存储输出结果
        for(int i=0;i < s.length();i++){
            if(s.charAt(i) >= '0' && s.charAt(i) <= '9'){
                sb.append("number");  //是数字,加number
            }
            else{
                sb.append(s.charAt(i));  //不是数字,加字母
            }
        }
        System.out.println(sb.toString());
        
    }
}

总结

1.思想

        题目的要求是输入获取一个字符串(只有小写字母和数字),保证其字母不变的同时,替换每一个数字为number,最后返回String。

2.算法流程

        首先用Scanner获取输入的String,并创建Stringbuild对象sb,用于存储返回的结果。然后,for循环遍历字符串String,判断元素如果是数字,就调用sb.append("number")将number添加到sb对象中,如果不是数组是字母,就把该字母append到sb对象中。最后输出sb对象即可。

第二次刷题总结

        直接pass了。

151.翻转字符串里的单词

题目

        给定一个字符串,逐个翻转字符串中的每个单词。

        示例 1:
        输入: "the sky is blue"
        输出: "blue is sky the"

        示例 2:
        输入: "  hello world!  "
        输出: "world! hello"
        解释: 输入字符串可以在前面或者后面包含多余的空格,但是反转后的字符不能包括。

        示例 3:
        输入: "a good   example"
        输出: "example good a"
        解释: 如果两个单词间有多余的空格,将反转后单词间的空格减少到只含一个。

代码

class Solution {
    public String reverseWords(String s) {
 
        StringBuilder sb = new StringBuilder();
 
        //先去除头尾和中间多余的空格
        sb = removeSpace(s);
 
        //翻转整个字符串
        sb.reverse();  //直接调用库函数翻转整个字符串
 
        //逐个翻转单个单词
        reverseEachWord(sb);
 
        return sb.toString();
    }
   
    //移除空格
    public StringBuilder removeSpace(String s){
        StringBuilder sb = new StringBuilder();
        int start = 0;
        int end = s.length() - 1;
        while(s.charAt(start) == ' '){  //去除首部多余空格
            start++;
        }
        while(s.charAt(end) == ' '){  //去除末尾多余空格
            end--;
        }
        for(int i = start;i <= end;i++){
            //如果当前start元素是字母,或者sb的最后一个元素不是空格,都要append
            if(s.charAt(i) != ' ' || sb.charAt(sb.length()-1) != ' '){
                sb.append(s.charAt(i));
            }
        }
        return sb;
    }
 
    //翻转每一个单词
    public void reverseEachWord(StringBuilder sb){
        int start = 0;  //用于表示某个单词的开始
        int end = 0;  //用于表示某个单词的结束
        //end会从到[0,sb.length()]遍历,关键是判断end是否是某个单词的结束索引
        for(;end <= sb.length();end++){
            //如果当前的end元素是空格,或者end已经指向了最后的空元素,说明end是结束索引
            //错误写法:sb.charAt(end) == ' ' || end == sb.length()
            if(end == sb.length() || sb.charAt(end) == ' '){
                reverse(sb,start,end-1);  //翻转单个单词,左闭右闭
                start = end + 1;   //start指向新的单词的第一个下标
            }
        }
    }
 
    //翻转函数
    public void reverse(StringBuilder sb,int i,int j){
        while(i < j){
            char tmp = sb.charAt(i);
            sb.setCharAt(i,sb.charAt(j));
            sb.setCharAt(j,tmp);
            i++;
            j--;
        }
    }
   
}

总结

1.思想

        这道题翻转字符串核心分为3个步骤。第一,去除头尾和单词中间的多余空格。第二,翻转整个字符串,先让单词的顺序与输出一致。第三,逐个翻转每个单词,让每个单词恢复原样。

2.算法流程

(1)如何去重重复空格

        对于String类型的s,用start指向0,end指向s.length()-1。首先,我们先去除头部的多余空格,即用while循环,只要当前的start元素是空格,就让start++。去除尾部同理,只要当前的end元素是空格,就让end--。这样操作完,start和end分别指向String中无首尾空格的索引位置。

        然后,我们要处理单词之间的多余空格,单保证词之间只保留一个空格。即用for循环从start遍历到end获取每一个元素,判断当前元素是不是我们要的字母or空格就行。那么如何判断呢?这里我们需要用StringBuilder保存我们需要的结果字符串,然后判断当前元素,如果是字母我们肯定要append,或者是两个单词中间的第一个空格我们也要append。如果不是,就不是我们要的,就让start向后继续遍历。

        因此,当前元素需要的判断条件就是(s.charAt(i) != ' ' || sb.charAt(sb.length()-1) != '  '),即如果当前元素是字母,或者我们的结果Stringbuilder对象sb的最后一个元素不是空格(这就说明当前元素是第一个空格了),我们就需要append到sb中。

(2)如何翻转字符串

        StringBuild对象用自带的reverse函数可以直接翻转整个字符串。或者自己写一个也行。

(3)如何翻转每个单词

        这里用到了双指针的思想。我们用start指向单词的首字母的索引,end表示指向的尾字母s索引+1。然后对字符串进行遍历,但这里要特别注意,[start,end)的区间表示一个完整的单词会有两种不同的情况,一种是end的元素==空格,那么[start,end)肯定是一个单词,可以翻转区间[start,end-1]的字符,另一种是end指向了sb.length(),因为最后一个单词后边没有空格,所以当满足条件end == sb.length()时,也要翻转[start,end-1]的字符。注意,翻转之后,要让start=end+1,即让start指向下一个单词的首字母的索引。

3.注意点

(1)翻转每一个单词的函数中,for循环遍历字符串时,循环的终止条件是end <= sb.length(),这里多一个等于,因为我们处理最后一个单词翻转的时候,end已经指向的sb.length(),不写等号的话,end的下标会越界。

(2)特别注意:end == sb.length() || sb.charAt(end) == ' ',这个判断条件,前后顺序千万不能错,因为如果把sb.charAt(end) == ' '写在前面,由于end最后会指向length(),导致charAt函数会出现下标越界。而把这个end == sb.length()写在前面,当end等于length时,该语句是true,||会直接截断处理,而不再判断后面的sb.charAt(end) == ' '语句。

4.语法问题

(1)StringBuilder对象的元素修改不能用=赋值,需要使用sb.setCharAt(修改元素的索引值,

新元素)进行设置。

第二次刷题总结

        直接通过了。

        移除空格的函数,需要先考虑头尾,然后从[start,end],判断只要当前元素是字母或者sb的最后不是空白,就可以append。

        翻转每个单词的函数稍微复杂一些,用的双指针。用[start,end)表示一个单词,因此特别注意end最后会指向字符串-1的位置,以及翻转的条件有两个,要么end是length,要么end位置的元素是空格,且end是length必须写前面。

  • 6
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

守岁白驹hh

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值