算法训练Day08 | LeetCode 151 反转字符串中的单词, 右旋字符串

LeetCode 151

151. 反转字符串中的单词 - 力扣(LeetCode)

思路

  头尾双指针法逆置所有单词
  1. 将字符串的逆转看作是字符串中单词的逆序,因此引入首尾的两个指针交换对应元素
  2. trim()去除头尾的空格
  3. 由于空格的长度不固定,此处我使用了正则表达式\\s+来表示任意长度的空格,并以此作为分割基准
  4. 分割后的数组交换头尾指针对应元素后再拼接在一起
  快慢指针法拼接单词
  1. 从字符串尾部开始,看作是一个单词一个单词的拼接
  2. trim()去除头尾空格
  3. 快指针找到第一个为空格的元素,此时说明fast指针刚好经历过了一个单词,那么此时将这个单词拼接在res中,记得加上" "
  4. 继续移动快指针,直到遇到不为空格的元素,说明此时已经过了单词间的空格情况,将慢指针移动到快指针的位置

注意点

  头尾双指针法逆置所有单词
  1. 只交换到字符串中间的位置,不要重复交换
  2. 交换后记得移动left,right指针
  3. 记得拼接" "
  快慢指针法拼接单词
  1. 使用substring的时候要注意它是一个左闭右开的函数,因此范围是[fast+1, slow+1]
  2. 最后的结果还要去一次头尾的空格

代码

  头尾双指针法逆置所有单词
class Solution {
    public String reverseWords(String s) {
        if (s == null) {
            return s;
        }

        StringBuffer res = new StringBuffer();
        s = s.trim();                            //去除头尾空格
        String[] tmp = s.split("\\s+");          //以空格为基准进行分割
        int left = 0;
        int right = tmp.length - 1;
        for (int i = 0; i < tmp.length; i++) {
            if(i < tmp.length / 2) {
                swap(left, right, tmp);
            }
            left++;
            right--;
            res.append(tmp[i]);
            res.append(" ");
        }
        res.deleteCharAt(res.length()-1);
        return res.toString();
    }

    public static void swap(int left, int right, String[] reTmp) {
            String t = reTmp[left];
            reTmp[left] = reTmp[right];
            reTmp[right] = t;
    }
}
  快慢指针法拼接单词
class Solution {
    public String reverseWords(String s) {
        if (s == null) {
            return s;
        }
        StringBuilder res = new StringBuilder();
        s = s.trim();
        int j = s.length() - 1;
        int i = j;
        while (i >= 0) {
            while (i >= 0 && s.charAt(i) != ' ')            //找到第一个空格处
                i--;
            res.append(s.substring(i + 1, j + 1) + " ");    //拼接单词
            while (i >= 0 && s.charAt(i) == ' ')            //找到不为空格处,删除单词间的空格
                i--;
            j = i;                                          //j移动到新单词尾部处
        }
        return res.toString().trim();
    }
}

右旋字符串

55. 右旋字符串(第八期模拟笔试) 

思路

把输出结果看作是两段字符串的拼接,分割点是字符串的长度-n

代码

import java.util.*;
class Main {
    public static void main(String[] args) {
        Scanner scan = new Scanner(System.in);
        int n = scan.nextInt();
        String s = scan.next();
        StringBuffer res = new StringBuffer();
        int length = s.length();
        res.append(s.substring(length-n, length));
        res.append(s.substring(0, length-n));
        System.out.println(res.toString());
    }
}

总结

今天的练习中,对于我来说印象最深刻的就是151的第二种解法,一是这是快慢指针的一个很妙的用法,二是这是一种新的解题想法,除了说可以正向的去思考,还可以逆向思考,就像这道题看作是倒着的单词拼接而不一定要是正向得思考如何变成逆序模样

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

TinaAmber

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

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

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

打赏作者

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

抵扣说明:

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

余额充值