算法题1LVPS|我们学校终于放暑假了...

不知道打开这一页的小伙伴内心世界是怎样的...
如题,我们学校终于开始放暑假了

微信图片_20180730170034.jpg

To 学生:嗷。。什么?你们早就放暑假了???那又怎样?我们爱学习,我们宁愿在图(qin)书(shi)馆(da)学(wang)习(zhe),对我们就是中国最晚放假的大学

To 已经毕业的coder:你点进来是何居心?工作中的bug调好了吗?业务逻辑理清楚了吗?blablabla


微信图片_20180730170109.png

等等,这和算法题有什么关系?
答案是

哦,没什么关系,就是日常闲得 dan teng 做做算法题,且看一道 LeetCode Hard 赛题

寻找最长合法括号子串
给定包含 '('')'的一个字符串, 找到一个最长的合法的括号子串。
Example 1:

Input: "(()"
Output: 2
解释: 最长有效括号子串是 "()"

Example 2:

Input: ")()())"
Output: 4
解释: 最长有效括号子串是 "()()"

这道题是考括号匹配算法,首先第一个想到的肯定是:暴力循环求解咯~

这里补充一下括号匹配算法:一般用 stack 解决,简单来说就是遇到 '(' 就push ‘)’就pop,如果pop失败或者最后stack非空就 return false.否则 return true.

显然匹配判断一下就是 O(N) 复杂度了

在做之前且慢,让我掐指一算 emmm 复杂度 O(N^3) 肯定不合算

当然这题我是想用动态规划来做的

while (想出方法){咬手指;}

然后我放弃了,因为手指真的好痛...于是我决定换一个姿势

while(想出方法){趴着打草稿;}

过了N久,对不起,恕小编无能,比不过各位大佬,这次我想不出睡着了...醒来翻翻solution 终于明白了

题解

首先,先上最简单的 Brute Force 算法

最简单暴力求解
// 匹配算法    
public boolean isValid(String s) {
        Stack<Character> stack = new Stack<Character>();
        for (int i = 0; i < s.length(); i++) {
            if (s.charAt(i) == '(') {
                stack.push('(');
            } else if (!stack.empty() && stack.peek() == '(') {
                stack.pop();
            } else {
                return false;
            }
        }
        return stack.empty();
    }
// 最长匹配求解
int main(){
int M=0;
for (int i=0;i<s.size();i++)
    for (int j=i+1;j<s.size();j++)
        if (isValid(s,i,j))
            M=max(M,j-i+1);
return 0;
}

时间复杂度 O(N^3)

其次,上了最高效的动态规划算法,看完答案都想自尽,因为太巧妙了...对不起,小编功底不到家...可是转念一想,不就是因为功底不到家才刷算法题提升自己能力的嘛...ヾ(≧▽≦*)o 于是我兴奋地坚持往下看了...

动态规划

dp[i] 表示以 i 下标结尾的匹配括号串的最长长度

那么当且仅当 s[i]=')'dp[i]\geq 0,这个时候才可能是合法的呀(●'◡'●)

  1. 如果 ".....() "则 dp[i]=dp[i-2]+2,这个式子很显然了

  2. 如果" .....)) "且 dp[i-dp[i-1]-1]='('dp[i]=dp[i-1]+dp[i-dp[i-1]-2]+2 表示分裂成两个相连的匹配括号串。

    解释一下 i-dp[i-1]-1 这个索引其实是以 dp[i-1] 为结尾的合法串的串头还要再往前一个的索引,因为 dp[i-1] 是以 s[i-1]=')' 为末尾的合法串长。那么如果 dp[i-dp[i-1]-1]='(',那么算 dp[i] 除了计算从 i-dp[i-1]-1='(' 为开头到 s[i-1]=')' 为末尾的串长,即 dp[i-1],我们还别忘了计算以 i-dp[i-1]-2 为末尾的合法串呀,虽然可能是 0

    找到 DP 方程编程什么的都是小 caaseee (●ˇ∀ˇ●)

    public int longestValidParentheses(String s) {
        int maxans = 0;
        int dp[] = new int[s.length()];
        for (int i = 1; i < s.length(); i++) {
            if (s.charAt(i) == ')') {
                if (s.charAt(i - 1) == '(') {
                    dp[i] = (i >= 2 ? dp[i - 2] : 0) + 2;
                } else if (i - dp[i - 1] > 0 && s.charAt(i - dp[i - 1] - 1) == '(') {
                    dp[i] = dp[i - 1] + ((i - dp[i - 1]) >= 2 ? dp[i - dp[i - 1] - 2] : 0) + 2;
                }
                maxans = Math.max(maxans, dp[i]);
            }
        }
        return maxans;
    }

时间复杂度 O(N)

再来一种贼巧妙的,至少我觉得是,给大家来拓宽一下思路~

别走,小心下一次面试说不定就碰到啦!

栈算法

使得当前最大长度 MaxLength 记录的始终是最长的合法的串

初始 stack.push(-1)

(1) 如果 s[i]='(' ,则 stack.push(i)

(2) 如果 s[i]=')', 则stack.pop()

stack.empty()

stack.push(i)

保证从 '(' 的前一个开始计

!stack.empty()

MaxLength=max\{MaxLength,i-stack.top()\}

保证 i-stack.top() 是合法串的长度

没有 code 感觉就像说废话一样,所以大家有空可以敲一敲

    public int longestValidParentheses(String s) {
        int maxans = 0;
        Stack<Integer> stack = new Stack<>();
        stack.push(-1);
        for (int i = 0; i < s.length(); i++) {
            if (s.charAt(i) == '(') {
                stack.push(i);
            } else {
                stack.pop();
                if (stack.empty()) {
                    stack.push(i);
                } else {
                    maxans = Math.max(maxans, i - stack.peek());
                }
            }
        }
        return maxans;
    }

这是第一篇 leetcode专题的文章,以后会出更多的~主要还是想一边自己学习的同时 和大家一起分享,一起进步,大家可能坐地铁或者什么的也不方便拿出电脑啊啥的,你只要打开文章以及纸笔就算是分析了一道题,也算是节约了时间了~

我当然比不过很多大牛,不过还是希望我包括大家都走在成为大牛的路上...

感谢大家敬请期待ヾ(≧▽≦*)o

个人微信公众号:准程序员coder
ID: wx_precoder
给各位准入程序员职场的发干货,日常算法题,机器学习等资料敬请期待~

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,718评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,683评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,207评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,755评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,862评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,050评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,136评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,882评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,330评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,651评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,789评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,477评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,135评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,864评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,099评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,598评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,697评论 2 351

推荐阅读更多精彩内容

  • 100个常用的javascript函数 1、原生JavaScript实现字符串长度截取 复制代码代码如下: fun...
    老头子_d0ec阅读 353评论 0 0
  • 正则表达式 为什么有正则表达式?因为查找是一个比较复杂的东西,我们更加准确快速查找自 己想要的 什么是正则表达式?...
    老头子_d0ec阅读 265评论 0 0
  • 目录管理 mkdir 创建一个目录 mkdir 目录名 •mkdir –p test1/test2/ 建立te...
    初见_0308阅读 388评论 0 0
  • 我的好心好意,为什么不接受 看见别人困难,会去帮上一把,其实也不是多大的事,可是过分干涉就变了味。 以前有...
    匿名J阅读 213评论 0 0
  • 用语言 用图画 用文字 用音乐 让爱流动 用眼晴 用嘴巴 用双臂 用两足 将爱表达
    迈一阅读 149评论 0 1