代码随想录算法训练营第八天 | LeetCode 344.反转字符串、541. 反转字符串II、剑指Offer 05.替换空格、151.翻转字符串里的单词、剑指Offer58-II.左旋转字符串

代码随想录算法训练营第八天 | LeetCode 344.反转字符串、541. 反转字符串II、剑指Offer 05.替换空格、151.翻转字符串里的单词、剑指Offer58-II.左旋转字符串

文章链接:代码随想录反转字符串        代码随想录反转字符串II        代码随想录剑指Offer 05.替换空格        代码随想录翻转字符串里的单词        代码随想录剑指Offer58-II.左旋转字符串

视频链接:代码随想录反转字符串        代码随想录反转字符串II        代码随想录翻转字符串里的单词

目录

代码随想录算法训练营第七天 | LeetCode 344.反转字符串、541. 反转字符串II、剑指Offer 05.替换空格、151.翻转字符串里的单词、剑指Offer58-II.左旋转字符串

1. LeetCode 344. 反转字符串

1.1 思路

1.2 代码

2. LeetCode 541. 反转字符串 II

2.1 思路

2.2 代码

3. 剑指Offer 05.替换空格

3.1 思路

3.2 代码

4. LeetCode 151. 反转字符串中的单词

4.1 思路

4.2 代码

5. 剑指Offer 58-II.左旋转字符串

5.1 思路

5.2 代码


1. LeetCode 344. 反转字符串

1.1 思路

  1. 根据题意很自然想到头尾双指针
  2. 头指针left=0;尾指针right=arr.length-1;定义临时变量temp进行交换
  3. while(left<right)这里有无等于号均可,交换以后两指针均往中间靠拢

1.2 代码

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

2. LeetCode 541. 反转字符串 II

2.1 思路

  1. 理清题意:够2k个字符,则前k个翻转;不够2k个也不够k个,则这部分全部翻转;不够2k个但够k个,则这部分不用翻转
  2. i遍历方法,由于这题是2k个2k个字符进行检验的,因此i+=2k,这样避免i++后用一个计数变量控制检验,避免了麻烦
  3. 如果符合2k长度的话,则翻转i到i+k,由于编程语言自己实现的函数一般都是左闭右开,因此i到i+k一般是不包含第i+k个的,因此这样就是翻转了k长度的字符串。这里需要判断一下if(i+k<=arr.length),不要超了数组的长度,这里需要等于号,因为假设i=0,k=3,剩余数组长度是3,符合翻转条件,这时i+k<=arr.length,可以翻转,说明要等于号
  4. 如果剩余数组不够k个了,那就将i到arr.length这部分直接翻转

2.2 代码

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

    public static void reverse(char[] ch,int i,int j){
        char temp;
        while(i<=j){
            temp=ch[i];
            ch[i]=ch[j];
            ch[j]=temp;
            i++;
            j--;
        }
    }
}

3. 剑指Offer 05.替换空格

3.1 思路

  1. 使用一个新的对象,复制 str,复制的过程对其判断,是空格则替换,否则直接复制,类似于数组复制
  2. 使用 StringBuilder 逐个复制 s ,遇到空格就append("%20"),否则就直接添加元素

3.2 代码

//使用一个新的对象,复制 str,复制的过程对其判断,是空格则替换,否则直接复制,类似于数组复制
public static String replaceSpace(String s) {
        if (s == null) {
            return null;
        }
        //选用 StringBuilder 单线程使用,比较快,选不选都行
        StringBuilder sb = new StringBuilder();
        //使用 sb 逐个复制 s ,碰到空格则替换,否则直接复制
        for (int i = 0; i < s.length(); i++) {
            //s.charAt(i) 为 char 类型,为了比较需要将其转为和 " " 相同的字符串类型
            //if (" ".equals(String.valueOf(s.charAt(i)))){}
            if (s.charAt(i) == ' ') {
                sb.append("%20");
            } else {
                sb.append(s.charAt(i));
            }
        }
        return sb.toString();
    }

4. LeetCode 151. 反转字符串中的单词

4.1 思路

  1. 整体思路是将整个字符串翻转,然后再将翻转后的单词逐个翻转
  2. 需要把开头、结尾、中间多余的空格删除,这里参考day01的移除元素这题
  3. 翻转字符串,参考day08的翻转字符串
  4. 翻转单词,同理参考day08的翻转字符串2

4.2 代码

class Solution {
   /**
     * 不使用Java内置方法实现
     * <p>
     * 1.去除首尾以及中间多余空格
     * 2.反转整个字符串
     * 3.反转各个单词
     */
    public String reverseWords(String s) {
        // System.out.println("ReverseWords.reverseWords2() called with: s = [" + s + "]");
        // 1.去除首尾以及中间多余空格
        StringBuilder sb = removeSpace(s);
        // 2.反转整个字符串
        reverseString(sb, 0, sb.length() - 1);
        // 3.反转各个单词
        reverseEachWord(sb);
        return sb.toString();
    }

    private StringBuilder removeSpace(String s) {
        // System.out.println("ReverseWords.removeSpace() called with: s = [" + s + "]");
        int start = 0;
        int end = s.length() - 1;
        while (s.charAt(start) == ' ') start++;
        while (s.charAt(end) == ' ') end--;
        StringBuilder sb = new StringBuilder();
        while (start <= end) {
            char c = s.charAt(start);
            if (c != ' ' || sb.charAt(sb.length() - 1) != ' ') {
                sb.append(c);
            }
            start++;
        }
        // System.out.println("ReverseWords.removeSpace returned: sb = [" + sb + "]");
        return sb;
    }

    /**
     * 反转字符串指定区间[start, end]的字符
     */
    public void reverseString(StringBuilder sb, int start, int end) {
        // System.out.println("ReverseWords.reverseString() called with: sb = [" + sb + "], start = [" + start + "], end = [" + end + "]");
        while (start < end) {
            char temp = sb.charAt(start);
            sb.setCharAt(start, sb.charAt(end));
            sb.setCharAt(end, temp);
            start++;
            end--;
        }
        // System.out.println("ReverseWords.reverseString returned: sb = [" + sb + "]");
    }

    private void reverseEachWord(StringBuilder sb) {
        int start = 0;
        int end = 1;
        int n = sb.length();
        while (start < n) {
            while (end < n && sb.charAt(end) != ' ') {
                end++;
            }
            reverseString(sb, start, end - 1);
            start = end + 1;
            end = start + 1;
        }
    }
}

5. 剑指Offer 58-II.左旋转字符串

5.1 思路

  1. 整体思路:局部翻转+整体翻转
  2. 先翻转区间为n的子字符串
  3. 再翻转区间为n到末尾的子字符串
  4. 最后翻转整个字符串

5.2 代码

class Solution {
    public String reverseLeftWords(String s, int n) {
        int len=s.length();
        StringBuilder sb=new StringBuilder(s);
        reverseString(sb,0,n-1);
        reverseString(sb,n,len-1);
        return sb.reverse().toString();
    }
     public void reverseString(StringBuilder sb, int start, int end) {
        while (start < end) {
            char temp = sb.charAt(start);
            sb.setCharAt(start, sb.charAt(end));
            sb.setCharAt(end, temp);
            start++;
            end--;
            }
        }
}
  • 3
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
第二十二天的算法训练主要涵盖了Leetcode题目中的三道题目,分别是Leetcode 28 "Find the Index of the First Occurrence in a String",Leetcode 977 "有序数组的平方",和Leetcode 209 "长度最小的子数组"。 首先是Leetcode 28题,题目要求在给定的字符串中找到第一个出现的字符的索引。思路是使用双针来遍历字符串,一个字符串的开头,另一个字符串的结尾。通过比较两个针所向的字符是否相等来判断是否找到了第一个出现的字符。具体实现的代码如下: ```python def findIndex(self, s: str) -> int: left = 0 right = len(s) - 1 while left <= right: if s[left == s[right]: return left left += 1 right -= 1 return -1 ``` 接下来是Leetcode 977题,题目要求对给定的有序数组中的元素进行平方,并按照非递减的顺序返回结果。这由于数组已经是有序的,所以可以使用双针的方法来解决问题。一个向数组的开头,另一个向数组的末尾。通过比较两个针所向的元素的绝对值的大小来确定哪个元素的平方应该放在结果数组的末尾。具体实现的代码如下: ```python def sortedSquares(self, nums: List[int]) -> List[int]: left = 0 right = len(nums) - 1 ans = [] while left <= right: if abs(nums[left]) >= abs(nums[right]): ans.append(nums[left ** 2) left += 1 else: ans.append(nums[right ** 2) right -= 1 return ans[::-1] ``` 最后是Leetcode 209题,题目要求在给定的数组中找到长度最小的子数组,

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值