给定一个仅包含 '('、')'、'{'、'}'、'['、']'的字符串,确定输入的字符串是否合法。

e.g. "()"、"()[]{}"、"[()]([]({}))" 是合法的,而"(]"、"([)]" 是不合法的。

使用栈stack

C++实现:

 bool isValid(string s) {
stack<char> stack;
for (char &c : s) {
if (c == '(')
stack.push(')');
else if (c == '{')
stack.push('}');
else if (c == '[')
stack.push(']');
else if (stack.empty() || (stack.top() != c))
return false;
else
stack.pop();
}
return stack.empty();
}

对应的Java实现:

 public boolean isValid(String s) {
Stack<Character> stack = new Stack<Character>();
for (char c : s.toCharArray()) {
if (c == '(')
stack.push(')');
else if (c == '{')
stack.push('}');
else if (c == '[')
stack.push(']');
else if (stack.isEmpty() || stack.pop() != c)  
return false;
}
return stack.isEmpty();
}

C++中的stack,其中有两个方法:

  • pop(),返回void,不返回类型<T>。
  • top(),返回栈顶的引用。

pop()不返回栈顶元素的原因是:

  • 异常安全原因

如果要实现,代码如下,

template<class T>
T stack<T>::pop() {
if( vused_ == ) {
throw "pop from empty stack";
} else {
return v_[--vused_ - ];
}
}

试图返回 v_[--vused_ - 1] 的时候,会调用 T 的拷贝构造函数,如果调用的时候发生异常,并没有返回正确的值,但是栈顶元素已经弹出了,这时就丢失了栈顶数据。

如果这样写,

template<class T>
T stack<T>::pop() {
if( vused_ == ) {
throw "pop from empty stack";
} else {
T result = v_[vused_ - ];
--vused_;
return result;
}
}

先建立一个栈顶元素的副本,如果拷贝构造函数出现异常,那么 --vused_ 不会执行,就不会丢失数据。但假如 --vused_ 和 return result 之间发生异常,依然会出问题。也就是说,如果想同时做 pop() 和 top() ,一定要保证 --vused_ 和 return result 是一个执行事务。

  • 效率原因

因为栈顶元素在栈中已经不存在,必须在按引用返回之前现将其存储到某个地方。如果选用动态内存,除非动态内存最终被删除,否则将导致内存泄露。

如果调用拷贝构造函数,对象按值传递的方式从函数返回,效率低下。

扩展 生成括号

给定 n 对括号,写出所有可能的合法括号组合。

e.g. 给定 n = 3,结果集为

[
  "((()))",
  "(()())",
  "(())()",
  "()(())",
  "()()()"
]

我的思路是递归

 vector<string> generateParenthesis(int n) {
if (n == ) {
return {"()"};
} else {
vector<string> result;
vector<string> pre = generateParenthesis(n - );
for (int i = ; i < pre.size(); i++) {
result.push_back("(" + pre[i] + ")");
result.push_back(pre[i] + "()");
if ("()" + pre[i] != pre[i] + "()")
result.push_back("()" + pre[i]);
}
return result;
}
}

算法没什么问题,可以生成所有可能的括号集合,但 vector 中存储的顺序不论怎么改都和答案的不一样。

答案使用的思路是DFS。创建一个函数来增加括号,str 表示加入到结果 vector<string> res 中的一项。

left 表示剩下的需要增加的“ ( ”的个数,初始为 n;right 表示当前的 str 中需要与没有配对的 “ ( ” 配对的 “ ) ” 的个数。

若 left > 0,则在 str 里加入一个“ ( ”,并让下次需要增加的“ ( ”的个数减 1,且因为加入了一个“ ( ”,该左括号没有右括号与之配对,所以当前 str 需要的“ ) ”加 1;

若 right > 0,说明需要在 str 里加入一个“ ) ”,下次需要的“ ) ”少 1 个就行了。

 vector<string> generateParenthesis(int n) {
vector<string> result;
addParen(result, "", n, );
return result;
} void addParen(vector<string> &res, string str, int left, int right) {
if (left == && right == ) {
res.push_back(str);
return;
}
if (left > )
addParen(res, str + "(", left - , right + );
if (right > )
addParen(res, str + ")", left, right - );
}

类似的,

这种方法个人觉得比上面的直观,一项 str 需要 n 个左括号和 n 个右括号,那么 left 和 right 都直接表示 str 所需要的“ ( ”和“ ) ”的个数。

注意的是,str 中一定要有没有配对的 “ ( ” 存在,才能加入 “ ) ”,所以加入右括号的判断条件是 if (right > left)。

 vector<string> generateParenthesis(int n) {
vector<string> result;
addParen(result, "", n, n);
return result;
} void addParen(vector<string> &res, string str, int left, int right) {
if (left == && right == ) {
res.push_back(str);
return;
}
if (left > )
addParen(res, str + "(", left - , right);
if (right > left)
addParen(res, str + ")", left, right - );
}

将 str 使用引用传参的话则不需要建立许多局部临时变量 str,可以节省一定的空间。

 vector<string> generateParenthesis(int n) {
vector<string> result;
string str = "";
addParen(result, str, n, n);
return result;
} void addParen(vector<string> &res, string &str, int left, int right) {
if (left == && right == ) {
res.push_back(str);
return;
}
if (left > ) {
str += "(";
addParen(res, str, left - , right);
str.resize(str.length() - );
}
if (right > left) {
str += ")";
addParen(res, str, left, right - );
str.resize(str.length() - );
}
}

注意 16 和 21 行 str.resize(str.length() - 1); 语句调用,由于 str 是引用,如果不重新设置 str 长度,str 会越来越长。

使用 pop_back 也可以

if (left > ) {
str.push_back('(');
addParen(res, str, left - , right);
str.pop_back();
}
if (right > left) {
str.push_back(')');
addParen(res, str, left, right - );
str.pop_back();
}

【LeetCode】Valid Parentheses合法括号的更多相关文章

  1. [LeetCode] 20. Valid Parentheses 合法括号

    Given a string containing just the characters '(', ')', '{', '}', '[' and ']', determine if the inpu ...

  2. [LeetCode] Valid Parentheses 验证括号

    Given a string containing just the characters '(', ')', '{', '}', '[' and ']', determine if the inpu ...

  3. [Leetcode] valid parentheses 有效括号对

    Given a string containing just the characters'(',')','{','}','['and']', determine if the input strin ...

  4. LeetCode Valid Parentheses 有效括号

    class Solution { public: void push(char c){ //插入结点 struct node *n=new struct node; n->nex=; n-> ...

  5. [leetcode]20. Valid Parentheses有效括号序列

    Given a string containing just the characters '(', ')', '{', '}', '[' and ']', determine if the inpu ...

  6. [LeetCode] 20. Valid Parentheses 验证括号

    Given a string containing just the characters '(', ')', '{', '}', '[' and ']', determine if the inpu ...

  7. [LeetCode] Generate Parentheses 生成括号

    Given n pairs of parentheses, write a function to generate all combinations of well-formed parenthes ...

  8. [LintCode] Valid Parentheses 验证括号

    Given a string containing just the characters '(', ')', '{', '}', '[' and ']', determine if the inpu ...

  9. LeetCode: Valid Parentheses 解题报告

    Valid Parentheses Given a string containing just the characters '(', ')', '{', '}', '[' and ']', det ...

随机推荐

  1. 8、nginx和tengine简介

    练习: 使用nginx反向代理(rr调度)用户请求至两个以上的后端LAMP(按标准路径部署的有pma,wd),不管用户请求是什么内容都反向代理至后端服务器去,但是如果用户请求的是图片或者是html,就 ...

  2. 用maven和spring搭建ActiveMQ环境

    前面搭建过了简单的环境,这次用稍微实际一点的maven+spring+activemq来进行搭建 准备:win7,eclipse,jdk1.8,tomcat8,maven3.5.2,activemq5 ...

  3. jquery事件重复绑定的几种解决方法 (二)

    防止事件重复绑定共有4种方法: bind().unbind()方法 live().die()方法 off().on()方法 one()方法 一.bind().unbind()方法 bind();绑定事 ...

  4. 使用ajax无法跨源问题总结

    参考文章: 浏览器同源政策及其规避方法 跨域资源共享 CORS 详解 使用jQuery实现跨域提交表单数据 <form action="http://v.juhe.cn/weather ...

  5. R语言网页爬虫

    R 是统计计算和数据分析的利器.给定一个数据集,利用前几章介绍到的 R 中灵活的数据结构或高性能计算,我们可以很方便地进行数据转换.建模和数值分析.一般来说,商业数据库会将数据以表格的形式很好地组织起 ...

  6. 《剑指offer》第六十七题(把字符串转换成整数)

    // 面试题67:把字符串转换成整数 // 题目:请你写一个函数StrToInt,实现把字符串转换成整数这个功能.当然,不 // 能使用atoi或者其他类似的库函数. #include <ios ...

  7. 《剑指offer》第四十三题(从1到n整数中1出现的次数)

    // 面试题43:从1到n整数中1出现的次数 // 题目:输入一个整数n,求从1到n这n个整数的十进制表示中1出现的次数.例如 // 输入12,从1到12这些整数中包含1 的数字有1,10,11和12 ...

  8. mui 访问系统相册将图片显示到网页

    访问系统相返回值为一个对象,通过转换为字符串可以查看,path.files[0]为返回路径去除路径赋值到src 调用摄像头返回的相片的path为一个路径通过 plus.io.resolveLocalF ...

  9. (转)c#反射

    1. 什么是反射2. 命名空间与装配件的关系3. 运行期得到类型信息有什么用4. 如何使用反射获取类型5. 如何根据类型来动态创建对象6. 如何获取方法以及动态调用方法7. 动态创建委托 1.什么是反 ...

  10. [转] @JoinColumn 详解 (javax.persistence.JoinColumn)

    原文链接:@JoinColumn详解  原文标的也是转载,但是没有注明原文链接,看起来乱乱的,所以整理一下转载过来,顺便细看一下 1. 一对一 现假设有Person表和Address表,是一对一的关系 ...