【LeetCode】151. 翻转字符串里的单词(剑指offer 58-I)
151. 翻转字符串里的单词
知识点:字符串;双指针
题目描述
给你一个字符串 s ,逐个翻转字符串中的所有 单词 。
单词 是由非空格字符组成的字符串。s 中使用至少一个空格将字符串中的 单词 分隔开。
请你返回一个翻转 s 中单词顺序并用单个空格相连的字符串。
说明:
输入字符串 s 可以在前面、后面或者单词间包含多余的空格。
翻转后单词间应当仅用一个空格分隔。
翻转后的字符串中不应包含额外的空格
示例
输入:s = "the sky is blue"
输出:"blue is sky the"
输入:s = " hello world "
输出:"world hello"
解释:输入字符串可以在前面或者后面包含多余的空格,但是翻转后的字符不能包括。
解法一:调用API
要对字符串,包括集合里的常用方法很熟悉,这样就可以直接调用里面的常用方法,比如说去掉首尾空格,比如说以某个字符分割,比方说以某个字符连接,比方说反转;
class Solution {
public String reverseWords(String s) {
//利用API;
s = s.trim();
String[] str = s.split("\\s+"); //正则表达式,根据一个或多个空格分割;
List<String> list = new ArrayList<>();
for(String i : str){
list.add(i); //将字符串加入列表;
}
Collections.reverse(list); //直接调用反转方法;
return String.join(" ", list); //字符串常用方法,以空格连接,返回一个字符串;
}
}
时间复杂度:O(N);
空间复杂度:O(N);
解法二:手写方法
我们上面主要使用了两个方法:去掉空格,反转;所以我们需要手动实现这两个方法;
- 去掉首尾两边和中间的多余空格;
- 将整个字符串反转;
- 将字符串里单个字符再反转;
需要注意的是,在java里字符串是不可变量,所以需要将其转化为可变的数据结构,比如数组;
class Solution {
public String reverseWords(String s) {
StringBuilder sb = trimSpaces(s);
reverse(sb, 0, sb.length()-1);
reverseWord(sb);
return sb.toString();
}
//去掉首尾和多余空格;
private StringBuilder trimSpaces(String s){
int left = 0, right = s.length()-1;
while(left <= right && s.charAt(left) == ' '){
left++;
} //去掉左侧空格,定位到左侧第一个无空格处;
while(left <= right && s.charAt(right) == ' '){
right--;
} //同理去掉右侧空格;
StringBuilder sb = new StringBuilder();
while(left <= right){
char c = s.charAt(left);
if(c != ' ') sb.append(c);
//到这证明遇到空格了,这时候看sb里最后一个不是空格就给加进去;
else if(sb.charAt(sb.length()-1) != ' ') sb.append(c);
left++;
}
return sb;
}
//反转整个字符;
private void reverse(StringBuilder sb, int front, int tail){
while(front < tail){
char temp = sb.charAt(front); //注意StringBuilder底层是数组,
//但是没有sb[front];获取指定索引就用charAt,包括String也是;
sb.setCharAt(front, sb.charAt(tail));
sb.setCharAt(tail, temp);
front++;
tail--;
}
}
//反转单个单词;
private void reverseWord(StringBuilder sb){
int start = 0, end = 0; //end记录每个单词的最后;
int l = sb.length();
while(start < l){
while(end < l && sb.charAt(end) != ' '){
end++;
} //end指定空格处;
reverse(sb, start, end-1);
start = end+1; //更新为下一个单词首处;
end++;
}
}
}
时间复杂度:O(N);
空间复杂度:O(N);
解法三:双指针
先把首尾空格去掉,然后定义两个指着,从后往前,分别指向每个单词的首和尾,再加入StringBuilder里;
class Solution {
public String reverseWords(String s) {
s = s.trim(); //去掉首尾空格;
StringBuilder sb = new StringBuilder();
int i = s.length()-1, j = i; //两个指着一个指向单词开头,一个指向结尾;
while(i >= 0){
while (i >= 0 && s.charAt(i) !=' ') i--; //i停在空格上;
sb.append(s.substring(i+1, j+1)+' '); //前开后闭;添加单词;
while (i >= 0 && s.charAt(i) == ' ') i--; //跳过空格;
j = i; //j指向下一个单词的尾;
}
return sb.toString().trim(); //因为上面在每个单词后面都加了空格,所以把最后的结尾的删除;
}
}
时间复杂度:O(N);
空间复杂度:O(N);
体会
- 要知道字符串的特点:在java语言中是不可变性。所以只要是需要修改字符串就需要引入新的可变的数据结构,常用StringBuilder。
- 要对String、StringBuilder常用方法熟悉;能想到。
相关链接
翻转字符串里单词
[剑指offer58-I 翻转单词顺序](https://leetcode-cn.com/problems/fan-zhuan-dan-ci-shun-xu-lcof/solution/mian-shi-ti-58-i-fan-zhuan-dan-ci-shun-xu-shuang-z/
【LeetCode】151. 翻转字符串里的单词(剑指offer 58-I)的更多相关文章
- C#版(击败100.00%的提交) - Leetcode 151. 翻转字符串里的单词 - 题解
版权声明: 本文为博主Bravo Yeung(知乎UserName同名)的原创文章,欲转载请先私信获博主允许,转载时请附上网址 http://blog.csdn.net/lzuacm. C#版 - L ...
- 【算法训练营day8】LeetCode344. 反转字符串 LeetCode541. 反转字符串II 剑指Offer05. 替换空格 LeetCode151. 翻转字符串里的单词 剑指Offer58-II. 左旋转字符串
[算法训练营day8]LeetCode344. 反转字符串 LeetCode541. 反转字符串II 剑指Offer05. 替换空格 LeetCode151. 翻转字符串里的单词 剑指Offer58- ...
- LeetCode 151. 翻转字符串里的单词(Reverse Words in a String)
151. 翻转字符串里的单词 151. Reverse Words in a String
- Java实现 LeetCode 151 翻转字符串里的单词
151. 翻转字符串里的单词 给定一个字符串,逐个翻转字符串中的每个单词. 示例 1: 输入: "the sky is blue" 输出: "blue is sky th ...
- LeetCode 151 翻转字符串里的单词
题目: 给定一个字符串,逐个翻转字符串中的每个单词. 示例 1: 输入: "the sky is blue" 输出: "blue is sky the" 示例 ...
- 微软面试题: LeetCode 151. 翻转字符串里的单词 出现次数:6
题目描述: 给定一个字符串,逐个翻转字符串中的每个单词. 说明: 无空格字符构成一个 单词 .输入字符串可以在前面或者后面包含多余的空格,但是反转后的字符不能包括.如果两个单词间有多余的空格,将反转后 ...
- leetcode python翻转字符串里的单词
# Leetcode 151 翻转字符串里的单词### 题目描述给定一个字符串,逐个翻转字符串中的每个单词. **示例1:** 输入: "the sky is blue" 输出: ...
- 代码随想录第八天 |344.反转字符串 、541. 反转字符串II、剑指Offer 05.替换空格 、151.翻转字符串里的单词 、剑指Offer58-II.左旋转字符串
第一题344.反转字符串 编写一个函数,其作用是将输入的字符串反转过来.输入字符串以字符数组 s 的形式给出. 不要给另外的数组分配额外的空间,你必须原地修改输入数组.使用 O(1) 的额外空间解决这 ...
- 力扣(LeetCode)翻转字符串里的单词 个人题解
给定一个字符串,逐个翻转字符串中的每个单词. 示例 1: 输入: "the sky is blue" 输出: "blue is sky the" 示例 2: 输 ...
随机推荐
- [源码解析] 深度学习分布式训练框架 horovod (2) --- 从使用者角度切入
[源码解析] 深度学习分布式训练框架 horovod (2) --- 从使用者角度切入 目录 [源码解析] 深度学习分布式训练框架 horovod (2) --- 从使用者角度切入 0x00 摘要 0 ...
- Spring Cloud10:Zipkin 服务跟踪
一.概述 为什么要有服务跟踪,分布式系统中有很多个服务在相互调用,调用关系是错综复杂的,如果这时出现了问题,我们在进行问题排查的时候,或者在优化架构的时候,工作量就比较大,这时候就需要我们能够准确的跟 ...
- UF_TRNS 变换相关
Open C uf5940uf5941uf5942 矩阵乘积变换uf5943 平移变换uf5944 缩放变换uf5945 旋转变换uf5946 镜像变换uf5947 实现变换,根据变换矩阵 ...
- 基于TensorFlow的服装分类
1.导包 #导入TensorFlow和tf.keras import tensorflow as tf from tensorflow import keras # Helper libraries ...
- TensorFlow入门实操课程第一章教程笔记
神经元网络深度学习的起步程序 Hello World 第一个应用程序总是应该从超级简单的东西开始,这样可以看到代码如何产生和运作的整体框架. 就创建神经网络而言,我喜欢使用的例子是一个能够学习两组数字 ...
- 「10.17-10.18」liu_runda’s模拟
暂咕 $day1$ A. 位运算 分类讨论,贡献分离. B. 集合论 维护类似时间戳的东西 C. 连连看 考场思路太局限了,考虑容斥. 我们可以看出两个方块能作出贡献,实际上是一个极大联通块(白块)所 ...
- 案例 | 腾讯广告 AMS 的容器化之路
作者 张煜,15年加入腾讯并从事腾讯广告维护工作.20年开始引导腾讯广告技术团队接入公司的TKEx-teg,从业务的日常痛点并结合腾讯云原生特性来完善腾讯广告自有的容器化解决方案 项目背景 腾讯广告承 ...
- vue项目中一些标签直接放在<template>下会报错Failed to compile with 1 errors
原因是a标签button以及element-ui的组件不能直接放在<template>下,需要先有一个div,其他标签要放在div下
- 看完互联网大佬的「LeetCode 刷题手册」, 手撕了 400 道 Leetcode 算法题
大家好,我是 程序员小熊 ,来自 大厂 的程序猿.相信绝大部分程序猿都有一个进大厂的梦想,但相较于以前,目前大厂的面试,只要是研发相关岗位,算法题基本少不了,所以现在很多人都会去刷 Leetcode ...
- Redis的主从数据一致性
我们学习了 AOF 和 RDB,如果 Redis 发生了宕机,它们可以分别通过回放日志和重新读入 RDB 文件的方式恢复数据,从而保证尽量少丢失数据,提升可靠性.不过,即使用了这两种方法,也依然存在服 ...