leetCode进阶算法题+解析(三十四)

打个广告,java技术交流群:130031711,欢迎大家踊跃加入!

最长上升子序列

题目:给定一个无序的整数数组,找到其中最长上升子序列的长度。

示例:
输入: [10,9,2,5,3,7,101,18]
输出: 4
解释: 最长的上升子序列是 [2,3,7,101],它的长度是 4。
说明:
可能会有多种最长上升子序列的组合,你只需要输出对应的长度即可。
你算法的时间复杂度应该为 O(n2) 。
进阶: 你能将算法的时间复杂度降低到 O(n log n) 吗?

思路:这个题有种似曾相识的感觉啊。但是我没明白题目的2 3 7 101。是指可以跳着?那2 5 7 101是不是也可以?再换个角度,是不是这样应该用dp来做?我反正直觉上觉得dp可以的。。。到第一位,最长串1,第二位最长串还是1,第三位,最长串1, 第四位最长串2。到第五串,最长串还是2.但是边界值变成了3,第六位变成了3.。。。我直觉是这样,具体怎么做还是要代码实现的,另外我记得之前一个类似的题目是有波线图,也不知道有没有用。我先去写代码试试了。
额,果然实现和想象有点不同,我随着做随着改思路,最终完全不是dp了。哈哈。先贴代码再说心路历程:

class Solution {
    public int lengthOfLIS(int[] nums) {
        if(nums.length==0) return 0;
        //这里用Integer如果不赋值是null
        int[] d = new int[nums.length];
        d[0] = Integer.MAX_VALUE;
        int lastIdx = 0;
        for(int i = 0;i<nums.length;i++){
            int n = nums[i];            
            if(n>d[lastIdx]){
                lastIdx++;
                d[lastIdx] = n;
            }else{
                for(int j = 0;j<=lastIdx;j++){
                    if(d[j]==n) break;//这个元素已经出现了直接pass,不然可能出现相同元素都占位
                    if(d[j]>n){
                        d[j] = n;
                        break;
                    }
                }
            }
        }
        return lastIdx+1;
    }
}

先说dp,dp是选和不选的问题,本来我的想法是两个比较点是前一个数的连续数和比当前数小的第一个数的个数+1。然后发现这里就需要遍历到比当前数小的第一个数的个数。
但是这样做的问题就是要不断往前遍历,甚至运气不好降序排列的话就是n方*n了。所以我做着做着发现可以换个思路,就是直接确定某个数可以是第几位的子序。并且如果2 5 7,这个时候出现了3,把5替换成3.变成了2,3,7。以后再有数字如果是大于7的,比如8,9,10 该往后续就往后续。但是如果是4,也可以续在23的后面,子序变成了2,3,4。这样哪怕后面再出现5,最开始的2 5 7是不能续了,但是这里的2,3,4就可以续上5,变成了2,3,4,5这样的子序。可以保证一次遍历下来获取的是最长子序。
其实时间复杂度也就那样,我感觉会比dp稍微好点。并且我上面的代码提交性能超过百分之百。0ms,所以就这样了?
这个题我觉得dp是没问题的,不然我再用dp做一次?就当是复习知识了。
dp实现了,反正性能不咋地,也不知道是我处理的有问题还是怎么样,但是我个人感觉dp性能不如上面的那个也是正常的,我直接贴代码:

class Solution {
    public int lengthOfLIS(int[] nums) {
        if(nums.length==0) return 0;
        int[] d = new int[nums.length];
        d[0] = 1;
        int res = 1;
        for(int i = 1;i<nums.length;i++){
            d[i] = 1;
            for(int j = 0;j<i;j++){
                if(nums[j]<nums[i]){
                    d[i] = Math.max(d[i],d[j]+1);
                }                
            }
            res = Math.max(res,d[i]);
        }
        return res;
    }
}

因为做的比较无脑,总体来讲就是一个元素所有比他小的都可以+1个子序。然后遍历前面的所有选择一个最大值来判断。其实这么想也是n方的实现。。。
题目说还能NlogN。一说log我就想起来二分法了。。。其实我感觉我第一种实现方式因为确定新建数组是排好序的了,确实可以用二分法或者双指针定位当前元素大小处于哪个位置。。不过我就懒得改了, 这道题做了一个多小时快两个小时了,我直接下一题了。

二维区域和检索-矩形不可变

题目:给定一个二维矩阵,计算其子矩形范围内元素的总和,该子矩阵的左上角为 (row1, col1) ,右下角为 (row2, col2)。上图子矩阵左上角 (row1, col1) = (2, 1) ,右下角(row2, col2) = (4, 3),该子矩形内元素的总和为 8。
题目截图

示例:
给定 matrix = [
[3, 0, 1, 4, 2],
[5, 6, 3, 2, 1],
[1, 2, 0, 1, 5],
[4, 1, 0, 1, 7],
[1, 0, 3, 0, 5]
]
sumRegion(2, 1, 4, 3) -> 8
sumRegion(1, 1, 2, 2) -> 11
sumRegion(1, 2, 2, 4) -> 12
说明:
你可以假设矩阵不可变。
会多次调用 sumRegion 方法。
你可以假设 row1 ≤ row2 且 col1 ≤ col2。

思路:如图所示,我觉得重点有两个:一个是矩形不可变,一个是这个方法会多次调用,所以我目前的想法是在初始化的时候就对这个矩形进行计算操作。然后每次调用这个方法的时候使得计算量大大的减少。至于怎么计算我再看看题。我目前的想法是横着压缩或者竖着压缩。然后真正判断的时候确定这个矩形是较短的边,然后两端相减再一层一层相加。肯定比现在这样全遍历要效果好,但是是不是最优解,是不是leetcode想要的解我不知道。我先去试试了。
额,反正实现是实现了,性能没及格,但是真的测试案例绝了,我先贴代码:

class NumMatrix {
    int[][] rows;
    boolean isLive;
    public NumMatrix(int[][] matrix) {
        if(matrix.length==0 || matrix[0].length==0) isLive = true; 
        if(!isLive){
            rows = new int[matrix.length][matrix[0].length];
            int r = 0;
            for(int i = 0;i<matrix.length;i++){           
                for(int j = 0;j<matrix[0].length;j++){
                    r += matrix[i][j];
                    rows[i][j] = r;
                }
                r = 0;
            }
        }
       
    }
    
    public int sumRegion(int row1, int col1, int row2, int col2) {
        if(isLive) return 0;
        int sum = 0;
        for(int i = row1;i<=row2;i++){
            if(col1==0){
                sum += rows[i][col2];
            }else{
                sum += rows[i][col2]- rows[i][col1-1];
            }            
        }
        return sum;
    }
}

/**
 * Your NumMatrix object will be instantiated and called as such:
 * NumMatrix obj = new NumMatrix(matrix);
 * int param_1 = obj.sumRegion(row1,col1,row2,col2);
 */

这个我觉得已经是简单的了。剩下的看性能排行第一的代码了:

class NumMatrix {

    int[][] dp;
    public NumMatrix(int[][] matrix) {
        if(   matrix           == null
       || matrix.length    == 0
       || matrix[0].length == 0   ){
        return;   
        }
        dp = new int[matrix.length + 1][matrix[0].length + 1];
        for (int i = 1; i < dp.length; i++) {
            for (int j = 1; j < dp[0].length; j++) {
                dp[i][j] = dp[i - 1][j] + dp[i][j- 1] - dp[i - 1][j - 1] + matrix[i - 1][j - 1];
            }
        }
    }
    
    public int sumRegion(int row1, int col1, int row2, int col2) {
        return dp[row2 + 1][col2 + 1] - dp[row1][col2 + 1] - dp[row2 + 1][col1] + dp[row1][col1];
    }   
}

人家第一不冤,我是单纯的一个方向压缩,人家是横竖都压缩了。但是这个题我感觉没啥难点,就是判断边界。剩下的都是思路了,先压缩应该很容易想到,我直接下一题了。

累加数

题目:累加数是一个字符串,组成它的数字可以形成累加序列。一个有效的累加序列必须至少包含 3 个数。除了最开始的两个数以外,字符串中的其他数都等于它之前两个数相加的和。给定一个只包含数字 '0'-'9' 的字符串,编写一个算法来判断给定输入是否是累加数。说明: 累加序列里的数不会以 0 开头,所以不会出现 1, 2, 03 或者 1, 02, 3 的情况。

示例 1:
输入: "112358"
输出: true
解释: 累加序列为: 1, 1, 2, 3, 5, 8 。1 + 1 = 2, 1 + 2 = 3, 2 + 3 = 5, 3 + 5 = 8
示例 2:
输入: "199100199"
输出: true
解释: 累加序列为: 1, 99, 100, 199。1 + 99 = 100, 99 + 100 = 199
进阶:
你如何处理一个溢出的过大的整数输入?

思路:先不说溢出。这个题我觉得重点是选择第一个数和第二个数。然后我打算用回溯来试试。首先第一位是第一个数,然后第二个数的位数从头判断到不溢出的尾。都不行就前两位是第一个数,然后继续那么判断。反正最终有true的就true,都遍历完还没true就是false。这个时间复杂度比较高,应该是n方。但是目前没有别的好思路了,我去写代码了。对了,题目说最少包含三个数,所以前两个数的和是第三个数,所以第二位数不用从头到尾,确定不会超过数组的一半到三分之一就行。然后第一个数也不会超过数组的二分之一。反正这个值比较不好算,我先去代码试试。多了也没事,只不过是多做无用功而已。
emmmm....终于过了!调试的死去活来的恶心~!我先贴代码;

class Solution {
    boolean flag;
    public boolean isAdditiveNumber(String num) {
        if(num.length()<3) return false;
        int n = num.length()/2;
        int m = (num.length()/3)*2;
        long a = 0l;
        long b = 0l;
        for(int i = 0;i<=n;i++){
            if(i>0 && num.charAt(0)=='0') break;
            a = a*10+(num.charAt(i)-'0');
            for(int j = i+1;j<=m;j++){
                if(j>i+1 && num.charAt(i+1)=='0') break;
                b = b*10+(num.charAt(j)-'0');
                if(isAdditive(num.substring(j+1),a,b) && flag) return true;
            }
            b = 0;
        }
        return false;
    }
    private boolean isAdditive(String remain, long num1, long num2) {
        if (remain.equals("")) return true;
        long sum = num1 + num2;
        String str = String.valueOf(sum);
        if (!remain.startsWith(str)) return false;
        flag = true;
        return isAdditive(remain.substring(str.length()), num2, sum);
    }
}

首先思路是对的,其次是这里不要直接把整个字符串拆成两部分!虽然这个条件判断差不多合理的,但是比如111这种只有三个数的时候,就会出问题。然后我加了个flag判断。
额,现在我发现可能是我范围取值不应该用小于等于。毕竟长度本身就比下标多了1 。改了一下,提交后还是对的,然后再贴一下代码:

class Solution {
    public boolean isAdditiveNumber(String num) {
        if(num.length()<3) return false;
        int n = num.length()/2;
        int m = (num.length()/3)*2;
        long a = 0l;
        long b = 0l;
        for(int i = 0;i<n;i++){
            if(i>0 && num.charAt(0)=='0') break;
            a = a*10+(num.charAt(i)-'0');
            for(int j = i+1;j<m;j++){
                if(j>i+1 && num.charAt(i+1)=='0') break;
                b = b*10+(num.charAt(j)-'0');
                if(isAdditive(num.substring(j+1),a,b)) return true;
            }
            b = 0;
        }
        return false;
    }
    private boolean isAdditive(String remain, long num1, long num2) {
        if (remain.equals("")) return true;
        long sum = num1 + num2;
        String str = String.valueOf(sum);
        if (!remain.startsWith(str)) return false;
        return isAdditive(remain.substring(str.length()), num2, sum);
    }
}

这个题思路蛮清楚的,就是细节多调试一下就行了,我也是面向测试案例不断调试,反正最后成功了!


提交截图

然后今天的笔记就记到这里,如果稍微帮到你了记得点个喜欢点个关注。也祝大家工作顺顺利利!生活健健康康!顺便打个广告:java技术交流群:130031711,欢迎大家踊跃加入!

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

推荐阅读更多精彩内容