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 ...
随机推荐
- C++ const使用总结
这里针对C++中const的一些一般用法进行一下简单的总结 一.定义常量 常量不可修改 : ; 与#define宏定义常量的区别:(1)const常量具有类型,编译器可以进行安全检查:#define宏 ...
- WPF图片,DataGrid等实现圆角
<Grid HorizontalAlignment="Center" VerticalAlignment="Center"> <Grid.Ro ...
- iota: Golang 中优雅的常量
阅读约 11 分钟 注:该文作者是 Katrina Owen,原文地址是 iota: Elegant Constants in Golang 有些概念有名字,并且有时候我们关注这些名字,甚至(特别)是 ...
- 深入C#并行编程(2) -- 使用线程
一.可以使用位于命名空间System.Threading中的Thrad类开启线程: //声明一个新的线程 Thread myThread = new Thread(LoopFunc); //传递一个T ...
- 工具类ToastUtil 避免在子线程中使用抛异常 "Can't create handler inside thread that has not called Looper.prepare()"
package com.example.kbr.utils; import android.view.Gravity; import android.widget.Toast; import io.r ...
- mssql sqlserver 使用sql脚本获取字符串存在多少个网址(url地址)的方法分享
摘要:下文讲述获取一个字符串中存在多少个网址的方法,如下实验环境:sql server 2008 R2 实现思路: 1.新建一个自定义函数,可将单个字符串拆分为含单个网址的数据表 2.采用outer ...
- [Linux] SSH隧道本地端口转发访问远程服务中的数据库
当我的本地没有安装任何数据库服务的时候,可以直接通过我本地的端口访问远程机器上的数据库服务,实现这样的效果就可以使用本地转发功能 实际测试本地端口转发,把本地的9006端口转发给远程服务器的115.1 ...
- (入门SpringBoot)SpringBoot项目事务(三)
Spring声明式事务的使用:由@Transactional进行标注,可以使用在类和方法上.当标注在类上,类下面所有公共非静态的方法都将启用事务功能.接下来,运行事务注解标注的方法,Spring的事务 ...
- 分布式系统ID的几种生成办法
前言 一般单机或者单数据库的项目可能规模比较小,适应的场景也比较有限,平台的访问量和业务量都较小,业务ID的生成方式比较原始但是够用,它并没有给这样的系统带来问题和瓶颈,所以这种情况下我们并没有对此给 ...
- 【洛谷P2494】 [SDOI2011]保密(分数规划+最小割)
洛谷 题意: 题意好绕好绕...不想写了. 思路: 首先类似于分数规划做法,二分答案得到到每个点的最小危险度. 然后就是在一个二分图中,两边撤掉最少的点(相应代价为上面算出的危险度)及相应边,使得中间 ...