题目大意:给出一个只包含字符'('和')'的字符串S,求最长有效括号序列的长度。

  


  很有趣的题目,有助于我们对这种人类自身制定的规则的深入理解,可能我们大多数人都从没有真正理解过怎样一个括号序列是有效的,因此解题也无从说起。整道题目的难度在于我们对有效括号序列的理解和定义。下面给出我自己的定义:、

  定义1:空括号序列是有效的。

  定义2:对于一对左右括号,若左括号出现在右括号的左边,且左右括号之间(不包含两端)的括号序列是有效的,那么称该左括号到该右括号(包含)这一段序列是有效的。且称该左括号和右括号匹配。

  定义3:若一个括号序列中每一个左括号都有与之匹配的右括号,而每一个右括号也有与之匹配的左括号,则称该括号序列是有效的。定义3是对定义2的一个扩展。

  对于定义的正确性与个人思维方式有关故不做说明,我一直认为无法证明是错误的东西,就是正确的,至少你能直接拿过来使用。下面给出一些记号的定义:S[x~y]表示由S[x],S[x+1],...,S[y-1],S[y]按先后顺序组成的子序列;n表示S的长度。后面不再对这些记号做说明。

  


  命题1:对于任意一个有效括号序列P,其中每一个左括号唯一匹配一个右括号,同时右括号也唯一匹配左括号。

  证明:定义3已经给出了有效序列中每个左括号与某个右括号匹配,只是缺少唯一性的证明,下面进行说明。假设对于某个下标为A的左括号同时下标为B与下标为C的两个右括号相匹配。不妨假设B<C。则由于A和B相匹配,由定义2可以得出序列P[A+1~C-1]是有效序列,而A+1<=B<=C-1,故能为找到下标为D的左括号,使得A+1<=D<B成立,且D与B想匹配(定义3)。可以重复上面的过程找到下标为E的右括号使得D<E<B,且D与E相匹配,从而无限推导下去,但是我们所考察的有效区间从A,C转为A,B,之后转为E,B,这是一个不断递减的过程,因此这个重复动作总会在某个点失败,而这时候就能反证假设错误,故唯一性得证。

  命题1':若P和P[x~y]均为有效序列,那么在P中下标为x的左括号必然与下标为y的右括号唯一匹配。

  证明:首先由于P[x~y]是有效序列,因此显然在P中下标为x的左括号与下标为y的右括号是匹配的。而由命题1即可直接得出匹配的唯一性。

  命题2:对于一段括号序列S[x~y]是有效的等价于:若对于任意满足x <= i <= y的整数i,有S[x~i]中左括号数不少于右括号数以及S[i~y]中右括号数不少于左括号数两个性质同时成立。

  证明:

  对于必要性,由于S[x~y]中每一个左括号都有对应的右括号与之唯一相匹配,因此对于任意满足x <= i <= y的整数i,有S[x~i]中左括号数不少于右括号数(否则S[x~i]中多余的右括号则在S[x~y]没有与之匹配的左括号)。而S[i~y]中右括号数不少于左括号数性质也可以类似推出。

  而对于充分性,利用数学归纳法。当y = x + 1时,命题显然成立,因为满足这一条件的子序列只可能是"()"。现假设y <= x + t时(t >= 1),命题成立。对于任意在S[x~y]之间出现的左括号,记其下标为L。由于S[L~y]中右括号不少于左括号数,因此必能找到最小的下标R,使得L<R且S[L~R]中出现的左括号数和右括号数数量一致。下面验证L和R是匹配的。首先L<R是不言而喻的,只要证明S[L+1~R-1]是有效的。由于下标R最小,因此可以得到对于任意满足L<m<R的整数m,都有S[L~m]中左括号数多于右括号数,否则与R的定义相悖。而同样能保证L是最大的满足L<R且S[L~R]中左括号与右括号数目一致的下标(利用反证法,设可以找到L<K<R使得S[K~R]中左括号与右括号数目一致,那么必然有S[L~K-1]中左括号与右括号数目一致),故对应的可以得出S[m~R]中右括号多于左括号数。利用这两个性质可以等价得出:对于任意满足L+1<=m<=R-1的整数m,都有S[L+1~m]中左括号数多于右括号数且S[m~R-1]中右括号多于左括号数。而(R-1)-(L+1)=R-L-2且y<=x+t+2可以推出R<=L+t,因此利用前面数学归纳法得到的结论(假设y <= x + t时命题成立)可以推出S[L+1~R-1]有效,即S[L~R]是有效的,再借助定义3可以得出序列S[x~y]是有效的。命题得证。

  

  命题3:若S[A~B]是有效序列,且S[B+1~C]是有效序列,则S[A~C]也是有效序列。

  证明:S[A~B]有效且S[B+1~C]是有效序列,则S[A~C]必然满足命题2右边的性质,故S[A~C]是有效序列。

  命题3':若S[A~C]是有效序列,且对于A<B<C,S[A~B]是有效序列,则S[B+1~C]是有效序列。

  证明:依旧是利用命题2右边的性质可以直接得出S[B+1~C]是有效序列。


  本来考虑的时候是希望能利用命题2给出的结论进行快速计算,后来发现动态规划似乎更加简单,也更容易证明。创建长度为n的数组DP,令DP[i]记录以下标i作为左边界的最大有效括号序列的长度,可以发现下面的递推关系:

  1.若S[i]是右括号,则DP[i]=0

  2.若S[i]是左括号,且S[i+1]是右括号,则DP[i]=2+DP[i+2]

  3.若S[i]是左括号,且S[i+1]是左括号,且S[DP[i+1]+i+1]是右括号,则DP[i]=DP[i+1]+2+DP[DP[i+1]+i+2]

  4.若S[i]是左括号,且S[i+1]是左括号,且S[DP[i+1]+i+1]是左括号,则DP[i]=0

  下面进行正确性的说明。

  1的正确性不言而喻。

  对于2,由于序列S[i~i+1+DP[i+2]]满足命题2右边的条件,因此序列有效。假设存在j > i + 1 + DP[i+2],使得S[i~j]有效,那么命题3'将导出S[i+2~j]是有效序列这样一个结论,这与DP的定义相悖。

  对于3,同样序列满足S[i~i+1+DP[i+1]]命题2右边的条件,因此序列有效。根据命题1',此时S[i]与S[DP[i+1]+i+1]唯一匹配。若我们忽略S[i+1~i+DP[i+1]]部分内容,则可以归结为情况2,利用相同的思路证明即可。

  对于4,能与左括号S[i]匹配的右括号的下标j只可能位于i+1与DP[i+1]+i+1(否则命题3'将导出与DP的定义相悖的结论)。而由于j不可能是DP[i+1]+i+1,因此j取i+1与DP[i+1]+i之间。但是由于i+1与DP[i+1]+i之间的任意下标为k的右括号,都必然会导出S[i~k]中左括号数多于右括号数目的现象,因此不存在与S[i]相匹配的右括号。

  利用上面的递推公式和动态规划的技术,可以在O(n)的时间复杂度和空间复杂度内得到最终结果,最长的有效序列的长度。


  下面给出实际代码:

 package cn.dalt.leetcode;

 /**
  * Created by dalt on 2017/8/26.
  */
 public class LongestValidParentheses {
     private static final char LEFT = '(';
     private static final char RIGHT = ')';

     public int longestValidParentheses(String s) {
         int n = s.length();
         char[] data = s.toCharArray();
         int[] dp = new int[n + 1];
         int max = 0;
         for (int i = n - 1; i >= 0; i--) {
             int nextIndex = i + 1;
             //condition 1
             if (data[i] == RIGHT) {
                 dp[i] = 0;
             }
             //check for not exceeding the bound
             else if (nextIndex >= n) {
                 dp[i] = 0;
             }
             //condition 2
             else if (data[nextIndex] == RIGHT) {
                 dp[i] = dp[i + 2] + 2;
             }
             //condition 3 and 4
             else if (data[nextIndex] == LEFT) {
                 int endIndex = dp[nextIndex] + nextIndex;
                 //check for not exceeding the bound
                 if (endIndex >= n || data[endIndex] == LEFT) {
                     dp[i] = 0;
                 }
                 //conditon 4
                 else {
                     //DP[i]=DP[i+1]+2+DP[DP[i+1]+i+2]
                     dp[i] = dp[nextIndex] + 2 + dp[endIndex + 1];
                 }
             }

             if (dp[i] > max) {
                 max = dp[i];
             }
         }
         return max;
     }
 }

  

leetcode: Longest Valid Parentheses分析和实现的更多相关文章

  1. [LeetCode] Longest Valid Parentheses 最长有效括号

    Given a string containing just the characters '(' and ')', find the length of the longest valid (wel ...

  2. [LeetCode] Longest Valid Parentheses 解题思路

    Given a string containing just the characters '(' and ')', find the length of the longest valid (wel ...

  3. [Leetcode] longest valid parentheses 最长的有效括号

    Given a string containing just the characters'('and')', find the length of the longest valid (well-f ...

  4. [LeetCode] Longest Valid Parentheses

    第一种方法,用栈实现,最容易想到,也比较容易实现,每次碰到‘)’时update max_len,由于要保存之前的‘(’的index,所以space complexity 是O(n) // 使用栈,时间 ...

  5. [LeetCode] Longest Valid Parentheses 动态规划

    Given a string containing just the characters '(' and ')', find the length of the longest valid (wel ...

  6. LeetCode: Longest Valid Parentheses 解题报告

    Longest Valid Parentheses Given a string containing just the characters '(' and ')', find the length ...

  7. [LeetCode] Longest Valid Parentheses -- 挂动态规划羊头卖stack的狗肉

    (Version 1.3) 这题在LeetCode上的标签比较有欺骗性,虽然标签写着有DP,但是实际上根本不需要使用动态规划,相反的,使用动态规划反而会在LeetCode OJ上面超时.这题正确的做法 ...

  8. leetcode Longest Valid Parentheses python

    class Solution(object): def longestValidParentheses(self, s): """ :type s: str :rtype ...

  9. [LeetCode] 032. Longest Valid Parentheses (Hard) (C++)

    指数:[LeetCode] Leetcode 指标解释 (C++/Java/Python/Sql) Github: https://github.com/illuz/leetcode 032. Lon ...

随机推荐

  1. C# #if, #else和#endif预处理指令

        #if 使您可以开始条件指令,测试一个或多个符号以查看它们是否计算为 true.如果它们的计算结果确实为true,则编译器将计算位于 #if 与最近的 #endif 指令之间的所有代码.例如, ...

  2. 使用c++实现一个FTP客户端(一)

    之前使用c++实现了一个FTP客户端,在这里做一些记录. 一.需要注意的几点 ①FTP是一种文件传输协议,基于TCP,所以客户端与服务器建立的连接是可靠.安全的,并且要经过三次握手的过程. ②FTP传 ...

  3. 解决navicat连接不上mysql服务器

    设置mysql密码 如果mysql用户密码为空,需要设置后第三方mysql操作工具才可以连接 进入数据库:mysql -uroot -p use mysql; update user set pass ...

  4. vmware克隆linux网络配置

    一.配置Linux网络 在安装Linux的时候,一定要保证你的物理网络的IP是手动设置的,要不然会在Linux设置IP连通网络的时候会报network is unreachable 并且怎么也找不到问 ...

  5. redis3.0自带集群配置

    参考 http://redis.readthedocs.org/en/latest/topic/cluster-tutorial.html http://yindashan.github.io/blo ...

  6. C# 如何捕获一个USB设备发送到PC的数据

    using System;using System.Collections.Generic;using System.ComponentModel;using System.Data;using Sy ...

  7. post参数的方法 json data 和特别的传参

    json格式传参: 那么久使用json的方式传参: json=payload data格式传参: 其他方式传参: 在webFormes里 value 的值不是普通的字符 要把value值先序列化在放入 ...

  8. VCS常用指令

    常用命令介绍 对VCS的常用命令进行介绍,便于工程师进行日常维护.本手册描述的命令仅供参考,具体描述请以Veritas公司提供的相关资料为准. VCS的安装和命令都在下列目录下:sbin, /usr/ ...

  9. istio 配置https gateway

    沒有親手實驗,参考官方文档:  https://istio.io/docs/tasks/traffic-management/secure-ingress/

  10. android中一个评分的控件

    RatingBar android中一个评分的控件 如何使用 Android Studio下: dependencies { compile 'com.hedgehog.ratingbar:app:1 ...