Longest Increasing Sequence
 public class Longest_Increasing_Subsequence {
     /**
      * O(N^2)
      * DP
      * 思路:
      *  示例:[1,0,2,4,10,5]
      *  找出以上数组的LIS的长度
      * 分析:
      *  只要求长度,并不要求找出具体的序列
      * 问题可以拆分为
      *      1. 对于[1],找出LIS
      *      2. 对于[1,0],找出LIS
      *      3. 对于[1,0,2],找出LIS
      *      4. 对于[1,0,2,4],找出LIS
      *      ...
      *      最后,对于[1,0,2,4,10,5],找出LIS
      * 再进一步思考,例如:
      *  找出[-1,0,1,0,2,4]的LIS,就要找到在4之前符合条件(都比4小且都为升序)的LIS的长度 => [-1,0,1]是满足情况的(最长,都是升序,都比4小)
      *  那么就要有一个数据结构来记录到某一个index上,LIS的长度。因为每一个index上的LIS长度并不是固定为前一个加1,所以每一个都要记录下来 => 数组dp[]
      *  dp[i]记录的是,在i这个index上,LIS的长度
      *  比如:
      * index 0 1 2 3 4 5
      *  dp:[ 1,2,3,1,4,5] //dp数组
      *  ar:[-1,0,1,0,2,4] //原数组
      *  dp[1] = 2表示在1这个index上,LIS的长度是2([-1,0])
      *  dp[4] = 4表示在4这个index上,LIS的长度是4([-1,0,1,2])
      * ----------------------------
      * 状态转换方程:
      * dp[i] = dp[k] + 1; (dp[k] = max(dp[0], dp[1], ... dp[i-1]))  // dp[i] = 在i以前最大的LIS长度加上1
      * 以上方程的成立条件:
      *      nums[k] < nums[i] //保持递增序列的属性
      */
     /**
      * O(N^2)
      */
     public int lengthOfLIS(int[] nums) {
         int[] dp = new int[nums.length];
         dp[0] = 1;
         for (int i = 1; i < nums.length; i++) {
             int beforeMaxLen = dp[i];
             // 在0 ~ i之间比较LIS的长度
             for(int j = 0; j < i; j++) {
                 if (nums[j] < nums[i] && dp[j] > beforeMaxLen) { //注意dp[j] > beforeMaxLen,新的长度要大于之前选出来的长度才能更新
                     beforeMaxLen = dp[j];
                 }
             }
             dp[i] = beforeMaxLen + 1;
         }
         int max = 0;
         // 在数组里找出最大的长度即可
         for (int i = 0; i < nums.length; i++) {
             if (dp[i] > max){
                 max = dp[i];
             }
         }
         return max;
     }
     /**
      * O(N*logN)
      * 思路:
      *  满足递增序列,就直接加入list中
      *  如果发现有降序出现,找出在原数组中比它大的第一个数的index,然后在list中替换那个数
      *  最后返回list的长度
      * 原理:
      *  因为只求长度,所以没有必要存储确切的sequence
      */
     public int lengthOfLIS_2(int[] nums) {
         List<Integer> list = new ArrayList<>();
         for(int num : nums) {
             if(list.isEmpty() || list.get(list.size() - 1) < num) { // 不满足递增序列
                 list.add(num);
             } else {
                 list.set(findFirstLargeEqual(list, num), num);
             }
         }
         return list.size();
     }
     private int findFirstLargeEqual(List<Integer> list, int target)
     {
         int start = 0;
         int end = list.size() - 1;
         while(start < end) {
             int mid = start + (end - start) / 2;
             if(list.get(mid) < target) {
                 start = mid + 1;
             }
             else {
                 end = mid;
             }
         }
         return end;
     }
     /**
      * 测试用
      */
     public static void main(String[] args) {
         Longest_Increasing_Subsequence lis = new Longest_Increasing_Subsequence();
         int[] a = {-1,0,1,0,2,4};
         System.out.print(lis.lengthOfLIS_2(a));
     }
 }
Longest Increasing Sequence的更多相关文章
- 动态规划 ---- 最长不下降子序列(Longest Increasing Sequence, LIS)
		分析: 完整 代码: // 最长不下降子序列 #include <stdio.h> #include <algorithm> using namespace std; ; in ... 
- [Leetcode] Binary search, DP--300. Longest Increasing Subsequence
		Given an unsorted array of integers, find the length of longest increasing subsequence. For example, ... 
- CSUOJ 1551 Longest Increasing Subsequence Again
		1551: Longest Increasing Subsequence Again Time Limit: 2 Sec Memory Limit: 256 MBSubmit: 75 Solved ... 
- [LeetCode] Binary Tree Longest Consecutive Sequence 二叉树最长连续序列
		Given a binary tree, find the length of the longest consecutive sequence path. The path refers to an ... 
- [LintCode] Longest Increasing Subsequence 最长递增子序列
		Given a sequence of integers, find the longest increasing subsequence (LIS). You code should return ... 
- The Longest Increasing Subsequence (LIS)
		传送门 The task is to find the length of the longest subsequence in a given array of integers such that ... 
- SPOJ LIS2 Another Longest Increasing Subsequence Problem 三维偏序最长链 CDQ分治
		Another Longest Increasing Subsequence Problem Time Limit: 20 Sec Memory Limit: 256 MB 题目连接 http://a ... 
- [LeetCode] Binary Tree Longest Consecutive Sequence II 二叉树最长连续序列之二
		Given a binary tree, you need to find the length of Longest Consecutive Path in Binary Tree. Especia ... 
- 673. Number of Longest Increasing Subsequence
		Given an unsorted array of integers, find the number of longest increasing subsequence. Example 1: I ... 
随机推荐
- 256. Paint House
			题目: There are a row of n houses, each house can be painted with one of the three colors: red, blue o ... 
- 实例学习写Makefile文件
			目录 0. 扫盲 1. 编译,链接 2. Makefile文件执行 3. Makefile书写规则 4. 案例 5. Makefile是如何工作的 6. 拔高,参考 0. 扫盲 Linux 环境下的程 ... 
- linux驱动学习之tasklet分析
			tasklet是中断处理下半部分最常用的一种方法,驱动程序一般先申请中断,在中断处理函数内完成中断上半部分的工作后调用tasklet.tasklet有如下特点: 1.tasklet只可以在一个CPU上 ... 
- 29 个 PHP 的 Excel 处理类
			下面的 PHP Excel 处理类中,包含 Excel 读写.导入导出等相关的类,列表如下: PHP Excel Reader classes 1. Read Excel Spreadsheets u ... 
- git cheat sheet,git四张手册图
- Python中模拟enum枚举类型的5种方法分享
			这篇文章主要介绍了Python中模拟enum枚举类型的5种方法分享,本文直接给出实现代码,需要的朋友可以参考下 以下几种方法来模拟enum:(感觉方法一简单实用) 复制代码代码如下: # way1 ... 
- TC SRM 593 DIV1 250(dfs)
			这图最多3色就可以 搜2就行了 #include <iostream> #include<cstdio> #include<cstring> #include< ... 
- Qt之启动外部程序
			简述 QProcess可以用来启动外部程序,并与它们交互. 要启动一个进程,通过调用start()来进行,参数包含程序的名称和命令行参数,参数作为一个QStringList的单个字符串. 另外,也可以 ... 
- UVa 11526 H(n)
			题意: long long H(int n){ long long res = 0; for( int i = 1; i <= n; i=i+1 ){ res = (res + n/i); } ... 
- HDU 4607 Park Visit (DP最长链)
			[题目]题意:N个城市形成一棵树,相邻城市之间的距离是1,问访问K个城市的最短路程是多少,共有M次询问(1 <= N, M <= 100000, 1 <= K <= N). [ ... 
