LeetCode 394:字符串解码 Decode String
题目:
给定一个经过编码的字符串,返回它解码后的字符串。
Given an encoded string, return its decoded string.
编码规则为: k[encoded_string],表示其中方括号内部的 encoded_string 正好重复 k 次。注意 k 保证为正整数。
The encoding rule is: k[encoded_string], where the encoded_string inside the square brackets is being repeated exactly k times. Note that k is guaranteed to be a positive integer.
你可以认为输入字符串总是有效的;输入字符串中没有额外的空格,且输入的方括号总是符合格式要求的。
You may assume that the input string is always valid; No extra white spaces, square brackets are well-formed, etc.
此外,你可以认为原始数据不包含数字,所有的数字只表示重复的次数 k ,例如不会出现像 3a 或 2[4] 的输入。
Furthermore, you may assume that the original data does not contain any digits and that digits are only for those repeat numbers, k. For example, there won't be input like 3a or 2[4].
示例:
s = "3[a]2[bc]", 返回 "aaabcbc".
s = "3[a2[c]]", 返回 "accaccacc".
s = "2[abc]3[cd]ef", 返回 "abcabccdcdcdef".
解题思路:
这道题类似我们之前做过的一道题:有效的括号: https://mp.weixin.qq.com/s/Sm1S26EgR-dC75hrhVnZGQ
只不过''有效的括号'' [] 内多了一些字符串需要操作。我们同样可以用数据结构栈来解题,,能用栈解决的题目大部分都可以用递归解决,两者逻辑基本相同:
输入:'3[a2[c]]'
初始化栈: 栈nums 存要重复的次数k,栈str 存字符串
遍历字符串:
指针指向字符'3',为数字
num暂存数字3
继续遍历,遇到字符'['
循环次数num入栈nums,空字符串res入栈str
nums: 3 res: ''
num置为0,str置空
继续遍历,遇到字符'a',为字母
空字符串res拼接字母'a',res='a'
继续遍历,遇到字符'2',为数字
num暂存数字2
继续遍历遇到字符'['
num入栈nums,res入栈str
nums: 3 -> 2 str: '' -> 'a'
num置为0,str置空
继续遍历,遇到字符'c',为字母
空字符串res拼接字母'c',res='c'
继续遍历遇到字符']'
nums弹出栈顶元素:当前字符串重复次数2
res = res*2 = 'cc'
str弹出栈顶元素'a'与res拼接并入栈:
res = 'a'+'cc'='acc'
str: '' -> 'acc'
继续遍历遇到字符']'
nums弹出栈顶元素:当前字符串重复次数3
res = res*3 = 'accaccacc'
str弹出栈顶元素空字符串''与res拼接并入栈:
res=''+'accaccacc'='accaccacc'
str: 'accaccacc'
结束返回res
注意:
- 由于重复次数可能大于10,所以暂存数字时要适当处理,如
num*10+当前数字 - 在c++里可以直接修改拼接字符,但Java不支持运算符重载,可以借助 StringBuilder 或 StringBuffer 类。
- 用栈暂存的逻辑与递归基本一致,可以理解为用递归实现栈。
- python没有栈这种数据结构,可以用 list() 数组或双端队列 deque()
- python可以只用一个栈以元组的形式重复次数和字符串,如
(num,res)
利用栈:
Java:
class Solution {
public String decodeString(String s) {
//初始化数据结构
Stack<StringBuilder> str = new Stack<>();
Stack<Integer> nums = new Stack<>();
StringBuilder res = new StringBuilder();
int num = 0;
for (char c : s.toCharArray()) {//递归字符串
if (c == '[') {
str.push(res);//入栈
nums.push(num);
num = 0;//刷新num、res
res = new StringBuilder();
} else if (c == ']') {
StringBuilder tmp = new StringBuilder();
for (int i = nums.pop(); i > 0; i--) tmp.append(res);//res*3
res = str.pop().append(tmp);
} else if (c >= '0' && c <= '9') num = num * 10 + (c - '0');//计算重复次数
else res.append(c);
}
return res.toString();
}
}
Python:
可直接操作字符串真的很方便。py里有现成的判断字符串的方法:
isdigit()是否为只包含数字的字符串isalpha()是否为只包含字母的字符串
class Solution:
def decodeString(self, s: str) -> str:
#初始化数据结构
stack, res, num = [], '', 0
for c in s:
if c.isdigit():
num = num * 10 + int(c)
elif c.isalpha():
res += c
elif c == '[':
#元组形式入栈
stack.append((res, num))
#刷新字符串和重复次数
res, num = '', 0
else:
#如果c==']',弹出字符串和重复次数
last_str, this_num = stack.pop()
res = last_str + this_num * res
return res
利用递归:
Java:
将 s.length() 的值以参数传递,减少重复调用 length() 造成的时间损耗
class Solution {
private int i = -1;//全局变量i,记录字符数组指针位置
public String decodeString(String s) {
return dfs(s.toCharArray(), s.length()).toString();
}
//递归函数
private StringBuilder dfs(char[] chars, int len) {
int num = 0;
StringBuilder str = new StringBuilder();
while (++i < len) {
if (chars[i] >= '0' && chars[i] <= '9')
num = num * 10 + (chars[i] - '0');
else if (chars[i] == '[') {
StringBuilder tmp = dfs(chars, len);//递归调用
while (--num >= 0) str.append(tmp);//重复字符串res=res*num
num = 0;
} else if (chars[i] == ']') return str;
else str.append(chars[i]);
}
return str;
}
}
Python:
class Solution:
i = -1
#递归函数,可以直接操作字符串就无需再建一个dfs辅助函数
def decodeString(self, s: str) -> str:
res, num = '', 0
while self.i < len(s) - 1:
self.i += 1
if s[self.i].isdigit():
num = num * 10 + int(s[self.i])
elif s[self.i].isalpha():
res += s[self.i]
elif s[self.i] == '[':
#递归调用
res += self.decodeString(s) * num
num = 0
elif s[self.i] == ']':
return res
return res
欢迎关注微.信公.众号:爱写Bug

LeetCode 394:字符串解码 Decode String的更多相关文章
- LeetCode 394. 字符串解码(Decode String) 44
394. 字符串解码 394. Decode String 题目描述 给定一个经过编码的字符串,返回它解码后的字符串. 编码规则为: k[encoded_string],表示其中方括号内部的 enco ...
- Java实现 LeetCode 394 字符串解码
394. 字符串解码 给定一个经过编码的字符串,返回它解码后的字符串. 编码规则为: k[encoded_string],表示其中方括号内部的 encoded_string 正好重复 k 次.注意 k ...
- [Swift]LeetCode394. 字符串解码 | Decode String
Given an encoded string, return it's decoded string. The encoding rule is: k[encoded_string], where ...
- [LeetCode]394. 字符串解码(栈)
题目 给定一个经过编码的字符串,返回它解码后的字符串. 编码规则为: k[encoded_string],表示其中方括号内部的 encoded_string 正好重复 k 次.注意 k 保证为正整数. ...
- [Leetcode]394.字符串解码
题目与解释 给定一个经过编码的字符串,返回它解码后的字符串. 编码规则为: k[encoded_string],表示其中方括号内部的 encoded_string 正好重复 k 次.注意 k 保证为正 ...
- Leetcode 394.字符串编码
字符串编码 给定一个经过编码的字符串,返回它解码后的字符串. 编码规则为: k[encoded_string],表示其中方括号内部的 encoded_string 正好重复 k 次.注意 k 保证为正 ...
- [LeetCode] 394. Decode String 解码字符串
Given an encoded string, return it's decoded string. The encoding rule is: k[encoded_string], where ...
- [LeetCode] Decode String 解码字符串
Given an encoded string, return it's decoded string. The encoding rule is: k[encoded_string], where ...
- [LeetCode] 271. Encode and Decode Strings 加码解码字符串
Design an algorithm to encode a list of strings to a string. The encoded string is then sent over th ...
随机推荐
- 融云技术分享:融云安卓端IM产品的网络链路保活技术实践
本文来自融云技术团队原创分享,原文发布于“ 融云全球互联网通信云”公众号,原题<IM 即时通讯之链路保活>,即时通讯网收录时有部分改动. 1.引言 众所周知,IM 即时通讯是一项对即时性要 ...
- Samba基础配置
本文环境:CentOS 7 简介 在UNIX-like之间共享文件系统主要是通过NFS实现的,而Windows之间共享文件系统主要是通过基于NetBIOS的网上邻居实现的,1984年Andrew Tr ...
- NumPy 学习 第二篇:索引和切片
数组索引是指使用中括号 [] 来定位数据元素,不仅可以定位到单个元素,也可以定位到多个元素.索引基于0,并接受从数组末尾开始索引的负索引. 举个例子,正向索引从0开始,从数组开始向末尾依次加1递增:负 ...
- WPF之动态加载曲线
首先说一下思路: 先创建一个控件(其实就是一个canvas),在canvas里面生成一条线,给这条线绑定一个PointCollection,在主界面中用一个定时器改变这个PointCollection ...
- Core源码(五)IQueryable(转)
如果要对Iqueryable进行深入的学习,那么对于拉姆达表达式.委托需要有一定了解.这里面拉姆达表达式就是匿名函数的一种写法,而委托我们常用FUNC和ACTION,这个都是对于delegate委托的 ...
- java基础(16):正则表达式、Date、DateFormat、Calendar
1. 正则表达式 1.1 正则表达式的概念 正则表达式(英语:Regular Expression,在代码中常简写为regex). 正则表达式是一个字符串,使用单个字符串来描述.用来定义匹配规则,匹配 ...
- ArchLinux 2019.11.01安装流程--安装基本系统
安装前的一些话 本文是参考官方文档ArchLinux的Installation guide(简体中文)加实际操作编写的. 有啥都好说,转载时请注明作者,这是基本素质,也是法律要求 安装是在虚拟机上进行 ...
- python 生成 树状结构
树状结构: 字典里只有一个键值对, key 为根, 值为一个列表, 列表里的某个或多个元素可以再进行分支(分支还是列表) 比如: 邮件的发件人, 收件人, 转发关系树状结构 forwarding_re ...
- 【微信小程序】App.js生命周期
1.小程序的生命周期-App.js App() 必须在 app.js 中注册,且不能注册多个.所以App()方法在一个小程序中有且仅有一个. App({ onLaunch: function () { ...
- C# 集合的交集 差集 并集 去重
C# 集合的交集 差集 并集 去重 两个对象list,直接比较是不行的,因为他们存的地址不一样 需要重写GetHashCode()与Equals(object obj)方法告诉电脑 class Stu ...