美文网首页动态规划
算法题1LVPS|我们学校终于放暑假了...

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

作者: 不会吃狗的猫不是兔子 | 来源:发表于2018-07-30 17:39 被阅读20次

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

微信图片_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
给各位准入程序员职场的发干货,日常算法题,机器学习等资料敬请期待~

相关文章

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

    不知道打开这一页的小伙伴内心世界是怎样的...如题,我们学校终于开始放暑假了 To 学生:嗷。。什么?你们早就放暑...

  • 算法题1LVPS

    且看一道 LeetCode Hard 赛题 寻找最长合法括号子串 给定包含 '(' 和 ')'的一个字符串, 找到...

  • 终于放暑假了

    小朋友早早就开始倒计时,什么时候开始放暑假,终于盼到暑假来了,老师的布置作业每天也是必须上交的。 单位也开始忙...

  • 周末的困难

    我们终于放暑假了,简直是太开心了,终于可以放松60天了。 不,...

  • 终于等到你~

    上学那时候,我们说话总喜欢用终于,终于考完试了,终于放暑假了,终于毕业了,终于离开这里了,终于过年了……仿佛任...

  • 薄荷味的暑假

    第一章 终于,放暑假了! 安菲菲:耶!太棒了,终于放暑假了! 叶古韵:安...

  • 学校解封了,却也放暑假了……

    5月伴着好消息的到来,学校要解除以往那种封闭式管理,但是陆陆续续的却有放假的通知,这是在五一假期发生的所有事情。在...

  • 我理想中的暑假生活

    我们终于放暑假了,好开心。 我的哥哥也放暑假了,我们可以一起玩耍,一起吃饭,一起写作业,一起打游戏,这...

  • 暑假!我不赞成给孩子报辅导班

    今天上初二的儿子放暑假, “爸,我们放暑假了” “嗯,你终于可以解脱了,可以好好的休息一下了” “爸爸,我假期...

  • 绘本讲师训练营[76期]9/21阅读原创《共读:大象巴巴之巴巴和

    76027 柳新颖 莎莉丝特城的大象学校在夏天的时候和我们一样也放暑假了,在大象王国上学的小猴子泽飞尔终于可...

网友评论

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

    本文链接:https://www.haomeiwen.com/subject/dcofvftx.html