贪心算法系列(二)|摆动序列最长递增子序列|买卖股票的最佳时机|买卖股票的最佳时机II

摆动序列

分析

  • 最经典的做法还是使用两个dp表的动态规划(代码放下面)
  • 这里采用贪心算法,直接上结论
  • 整个序列中,波峰波谷+起点和重点的个数 就是整个最长的摆动序列长度

    那么如何判断波峰/波谷呢?也很简单
  • left = nums[i] - nums[i-1]
  • right = nums[i+1] - nums[i]
  • 波峰/波谷:left * right < 0
  • 其余位置:left * right > 0

如果趋势是平的怎么处理

后面两种情况从宏观角度来看是存在波峰和波谷的,所有相同的点(平直部分)其实可以当作一个点,所以如果遇到right == 0,直接跳过即可

动态规划代码

java 复制代码
class Solution {
    public int wiggleMaxLength(int[] nums) {
        int n = nums.length;

        int[] f = new int[n], g = new int[n];
        for(int i = 0; i < n; i++) f[i] = g[i] = 1;

        int ret = f[0];
        for(int i = 1; i < n; i++) {
            for(int j = 0; j < i; j++) {
                if(nums[i] - nums[j] > 0) f[i] = Math.max(g[j] + 1, f[i]);
                else if(nums[i] - nums[j] < 0) {g[i] = Math.max(f[j] + 1, g[i]);}
            }

            ret = Math.max(f[i], g[i]);
        }

        return ret;
    }
}

贪心算法代码

java 复制代码
class Solution {
    public int wiggleMaxLength(int[] nums) {
        int left = 0, right = 0, ret = 0;
        for(int i = 0; i < nums.length - 1; i++) {
            right = nums[i + 1] - nums[i];// 右边趋势
            if(right == 0) continue;// 平地  直接略过即可
            if(left * right <= 0) ret++;// 波峰/波谷
            left = right;
        }

        return ret + 1;// 结尾也算  因为可升可降
    }
}

最长递增子序列
动态规划的解题思路中,当走到i位置的元素时,我们只关心i位置之前的所有递增子序列的最后位置的元素nums[j]是否小于nums[i],如果小于,就能构成一个新的,更长的递增子序列,取所有情况中的最大值

观察dp的思路不难发现,只需要关心递增子序列最后位置的元素的大小即可,无需关注序列本身的组成,

dp解法

java 复制代码
class Solution {
    public int lengthOfLIS(int[] nums) {
        // 时间复杂度O(N^2)
        int n = nums.length;
        int[] dp = new int[n];
        for (int i = 0; i < n; i++)
            dp[i] = 1;

        int ret = 1;
        for (int i = 1; i < n; i++) {
            for (int j = 0; j < i; j++) {
                if (nums[i] > nums[j])
                    dp[i] = Math.max(dp[i], dp[j] + 1);
            }
            ret = Math.max(ret, dp[i]);
        }

        return ret;
    }
}

思路

贪心+二分解法

java 复制代码
class Solution {
    public int lengthOfLIS(int[] nums) {
        // 时间复杂度O(N * logN)
        int n = nums.length;
        List<Integer> ret = new ArrayList<>();
        ret.add(nums[0]);

        for(int i = 1; i < n; i++) {
            // 处理边界情况
            if(nums[i] > ret.get(ret.size() - 1)) {
                ret.add(nums[i]);
            }else {
                // 不是边界--二分搜索插入
                int l = 0, r = ret.size();
                while(l < r) {
                    int mid = l + ((r - l) >> 1);
                    if(ret.get(mid) >= nums[i]) r = mid;
                    else l = mid + 1;
                }
                ret.set(l, nums[i]);
            }
        }
        return ret.size();
    }
}

相似题目(简化版)

链接:递增的三元子序列

java 复制代码
class Solution {
    public boolean increasingTriplet(int[] nums) {
        int len = nums.length;
        if(len < 3) return false;

        // min就是len==1的递增子序列的最后位置的元素
        // mid就是len==2的递增子序列的最后位置的元素
        // 只要找到比mid大的,就存在长度为三递增子序列
        int min = Integer.MAX_VALUE, mid = Integer.MAX_VALUE;
        for(int x : nums) {
            if(x <= min) min = x;
            else if(x > min && x <= mid) mid = x;
            else return true;
        }

        return false;
    }
}

买卖股票的最佳时机

贪心策略: 每次都假定今天卖出,那么就需要知道今天之前的最小值,这样才能获得当天的最大利润,所有的今天构成了每天 ,这样遍历一遍数组就能得到最大利润

java 复制代码
class Solution {
    public int maxProfit(int[] prices) {
        if(prices == null || prices.length == 0) return 0;
        int minPrice = Integer.MAX_VALUE, maxProfit = 0;

        for(int price : prices) {
            if(price < minPrice) minPrice = price;// 记录最低价格
            if(price - minPrice > maxProfit) maxProfit = price - minPrice;// 记录最大利润
        }

        return maxProfit;
    }
}

买卖股票的最佳时机II

  • 贪心算法:每一步都在做当前认为最优的操作


代码

java 复制代码
class Solution {
    public int maxProfit(int[] prices) {
        // 只要是上升趋势就卖出
        int ret = 0;
        for(int i = 1; i < prices.length; i++)
            if(prices[i] > prices[i - 1])
                ret += prices[i] - prices[i - 1];

        return ret;
    }
}
相关推荐
职略1 小时前
负载均衡类型和算法解析
java·运维·分布式·算法·负载均衡
A22741 小时前
LeetCode 196, 73, 105
java·算法·leetcode
阿里巴巴P8资深技术专家2 小时前
Java常用算法&集合扩容机制分析
java·数据结构·算法
zengson_g3 小时前
当需要对大量数据进行排序操作时,怎样优化内存使用和性能?
java·数据库·算法·排序算法
爱上电路设计3 小时前
有趣的算法
开发语言·c++·算法
Kerry_65 小时前
2024年江苏省研究生数学建模科研创新实践大赛C题气象数据高精度融合技术研究论文和代码分析
算法·数学建模·matlab·数据分析
风啊雨6 小时前
刷题Day44|188.买卖股票的最佳时机IV、309.最佳买卖股票时机含冷冻期、714.买卖股票的最佳时机含手续费
算法
yachihaoteng6 小时前
Studying-代码随想录训练营day27| 贪心算法理论基础、455.分发饼干、376.摆动序列、53.最大子序和
c++·算法·leetcode·贪心算法
逸群不凡6 小时前
C++|哈希应用->布隆过滤器
开发语言·数据结构·c++·算法·哈希算法
LNTON羚通6 小时前
视频共享融合赋能平台LnyonCVS国标视频监控平台包含哪些功能
大数据·网络·人工智能·算法·音视频