[LeetCode] Decode String 题解
题目
s = "3[a]2[bc]", return "aaabcbc".
s = "3[a2[c]]", return "accaccacc".
s = "2[abc]3[cd]ef", return "abcabccdcdcdef".
大概意思是根据指定的格式将字符串进行展开
思路
- 递归的思路
递归计算出括号最里面的字符串,依次再处理外面一层的字符串,每个单元内的字符串类似于一个结点,个数则为结点的个数。父结点则是将这些个数的字符串组合在一起,以此类推到跟结点,就是我们要求的结果。
- 迭代的思路
用两个栈来分别保存下单元中的个数,另一个则保存单元中的字符串,注意的是,要将最新的处理完后的字符串加入到栈中。一直加入,直到最后返回栈顶字符串则为所求结果。
实现
//
//
#include "../PreLoad.h"
/*
s = "3[a]2[bc]", return "aaabcbc".
s = "3[a2[c]]", return "accaccacc".
s = "2[abc]3[cd]ef", return "abcabccdcdcdef".
*/
class Solution {
public:
// 错误做法
string decodeString(string s) {
if (s == "") {
return "";
}
string result = "";
stack<pair<char, int>> codes;
codes.push({s[0], 0});
while (!codes.empty()) {
auto content = codes.top();
codes.pop();
// 表明开始是数字
if (content.first >= '0' && content.first <= '9') {
int start = content.second;
string nums = "";
nums.push_back(content.first);
while (s[start] != '\n' && s[start] != '[') {
content.first += s[start];
start++;
}
// 计算单位的字符串
int num = atoi(nums.c_str());
string str = "";
for (int i = 0; i < num; i++) {
str += content.first;
}
result += str;
// 找到下一个单位起始的位置
while (s[start] != '\n' && s[start] != ']') {
start++;
}
if (start != s.size()-1) {
start++;
codes.push({s[start], start});
}
}
}
return result;
}
/**
* 迭代DFS的做法
*
* @param s
* @return
*/
string decodeString2(string s) {
int cnt = 0; //每个单元里的数字个数
string content = "";
int len = s.size();
int i = 0;
stack<string> strs;
stack<int> cnts;
while (i < len) {
// 为数字的情况
if (i < len && s[i] >= '0' && s[i] <= '9') {
cnt = cnt * 10 + (s[i] - '0');
}
// 为左括号的情况
else if (i < len && s[i] == '[') {
cnts.push(cnt);
strs.push(content); //因为存在内嵌的情况,所以保存下之前的单元内的字符串
cnt = 0;
content.clear(); //清空用来保存后面单元的字符串
}
// 为右括号的情况
else if (i < len && s[i] == ']') {
int cnt_temp = cnts.top();
cnts.pop();
while (cnt_temp--) {
strs.top() += content;
}
// 将之前的字符串进行累加以后得到新的单元的内容
content = strs.top();
strs.pop();
}
// 中间的字符串
else {
content += s[i];
}
i++;
}
// 可能存在没有]的单元
return strs.empty() ? content : strs.top();
}
/**
* 递归DFS
* 计算单元格内字符串的内容,再根据个数进行累加
* 注意字符串下标和边界条件
*
* @param s
* @return
*/
string decodeString3(string s) {
int len = s.size();
if (len == 0) {
return "";
}
int i = 0;
return decodeHelper(s, i);
}
string decodeHelper(string s, int& i) {
string result = "";
int len = s.size();
while (i < len && s[i] != ']') {
// 非数字而是字符的情况下
if (s[i] < '0' || s[i] > '9') {
result += s[i++];
}
else {
// 计算数字
int cnt = 0;
while (s[i] >= '0' && s[i] <= '9' && i < len) {
cnt = cnt * 10 + (s[i++] - '0');
}
// 移动到字符串开始的位置
i++;
// 获得单元内的字符串,此时递归返回i的位置是]的位置
string str = decodeHelper(s, i);
//移动到下一个单元的位置
i++;
while (cnt--) {
result += str;
}
}
}
return result;
}
void test() {
string str = "3[a2[c]]";
cout << "result : " << decodeString3(str) << endl;
}
};
总结:这些和在树中使用DFS的思想是类似的。
[LeetCode] Decode String 题解的更多相关文章
- [LeetCode] Decode String 解码字符串
Given an encoded string, return it's decoded string. The encoding rule is: k[encoded_string], where ...
- LeetCode——Decode String
Question Given an encoded string, return it's decoded string. The encoding rule is: k[encoded_string ...
- [LeetCode] 394. Decode String 解码字符串
Given an encoded string, return it's decoded string. The encoding rule is: k[encoded_string], where ...
- LeetCode 394. 字符串解码(Decode String) 44
394. 字符串解码 394. Decode String 题目描述 给定一个经过编码的字符串,返回它解码后的字符串. 编码规则为: k[encoded_string],表示其中方括号内部的 enco ...
- [LeetCode] Encode String with Shortest Length 最短长度编码字符串
Given a non-empty string, encode the string such that its encoded length is the shortest. The encodi ...
- [LeetCode] Decode Ways 解题思路
A message containing letters from A-Z is being encoded to numbers using the following mapping: 'A' - ...
- LeetCode——Reverse String
LeetCode--Reverse String Question Write a function that takes a string as input and returns the stri ...
- Leetcode 8. String to Integer (atoi) atoi函数实现 (字符串)
Leetcode 8. String to Integer (atoi) atoi函数实现 (字符串) 题目描述 实现atoi函数,将一个字符串转化为数字 测试样例 Input: "42&q ...
- Leetcode -- 394. Decode String
Given an encoded string, return it's decoded string. The encoding rule is: k[encoded_string], where ...
随机推荐
- 学习篇之String()
// 3个特殊的引用类型:Boolean,Number,String var s1 = "some text"; ,); // me t ,); // me ,-); // so ...
- Python2和Python3中除法操作/的不同
X/Y 在3.0版本之前的Python中 >>>1/2 0 即一个整数(无小数部分的数)被另外一个整数除,计算结果的小数部分被截除了,只留下了整数部分 有时候,这个功能比较有用,譬如 ...
- 使用Dom解析器,操作XML里面的信息
import java.io.FileNotFoundException;import java.io.FileOutputStream;import java.io.IOException;impo ...
- VS error 全集(error C2664: 'CWnd::MessageBoxW' : cannot convert parameter 1 from 'char *' to 'LPCTSTR'的解决方法)
我用的是VS2005,在编译MFC时遇到了如下错误: error C2664: 'CWnd::MessageBoxW' : cannot convert parameter 1 from 'char ...
- Win32/MFC/COM学习推荐书籍
以前有不少朋友问关于学习各种技术的推荐书籍的问题,这里把我觉得比较好的一些书籍列一下,希望能起到抛砖引玉的作用就好了:) Win32开发 Programming Windows by Charles ...
- android shell常用命令
du -sm foldername 查看文件夹foldername 的大小,单位是兆(m),du -sk foldername单位是k adb shell rm -r /mnt/sdcard/ ...
- 读书笔记 effective c++ Item 7 在多态基类中将析构函数声明为虚析构函数
1. 继承体系中关于对象释放遇到的问题描述 1.1 手动释放 关于时间记录有很多种方法,因此为不同的计时方法创建一个TimeKeeper基类和一些派生类就再合理不过了: class TimeKeepe ...
- 学习笔记——Java数字处理类
1.数字格式化 使用Java.text.DecimalFormat格式化数字,一般使用其中的DecimalFormat类.如: import java.text.DecimalFormat; publ ...
- 初识html5——试试博文编辑器
1.html5简介 HTML5 将成为 HTML.XHTML 以及 HTML DOM 的新标准. HTML 的上一个版本诞生于 1999 年.自从那以后,Web 世界已经经历了巨变. HTML5 仍处 ...
- 【转】jqGrid学习之安装
jqGrid安装很简单,只需把相应的css.js文件加入到页面中即可. 按照官网文档: /myproject/css/ ui.jqgrid.css /u ...