算法整理-动态规划和Two Pointers

一 .   最长子序列和

令dp[i] 为以i结尾的最长子序列和。dp[i] = max(dp[i-1] + nums[i], nums[i])。 同时纪录dp[i]遍历结果的中的最大值。需要三个变量,纪录上一个dp,  当前dp和最大的dp.

class Solution {
public:
    int maxSubArray(vector<int>& nums) {
        int len = nums.size();
        if (len == 0) return 0;
        int pre = nums[0];
        int re;
        int max_re = pre;
        for (int i = 1; i < len; i++) {
            if (nums[i] > pre + nums[i]) re = nums[i];
            else re = pre + nums[i];
            pre = re;
            if (re > max_re) max_re = re;
        }
        return max_re;
    }
};

二. House RobberI

只有dp(n) = max(dp(n-1), notakedp(n-1) + nums[i]))

class Solution {// DP
public:  // dp(n) = max(dp(n-1), notakedp(n-1) + nums[i]) 第i个没抢, 抢了
    int rob(vector<int>& nums) {
        int take = 0;
        int notake = 0;
        int dp = 0;
        for(int i = 0; i < nums.size(); i++) {
            take = notake + nums[i];
            notake = dp;
            dp = max(take, notake);
        }
        return dp;
    }
};

更难的问题。是一个环,第一家和最后一家也不能抢。则这个问题可以求解为0,1, .. n-1, 的问题和1....n的问题,再求这个两个子数组情况下的最大的最优解。

三.  Maximal Square

https://www.cnblogs.com/grandyang/p/4550604.html

动态规划问题,dp[i][j]表示以matrix[i][j]为右下角的最大的square的边长。则dp[i][j] = min(dp[i-1][j-1], dp[i][j-1], dp[i-1][j]) + 1;。。当然只有matrix[i][j]为‘1’时这个公式才生效。

class Solution {
public:
    int maximalSquare(vector<vector<char>>& matrix) {
        if (matrix.empty() || matrix[0].empty()) return 0;
        int m = matrix.size(), n = matrix[0].size(), res = 0;
        vector<vector<int>> dp(m, vector<int>(n, 0));
        for (int i = 0; i < m; ++i) {
            for (int j = 0; j < n; ++j) {
                if (i == 0 || j == 0) dp[i][j] = matrix[i][j] - ‘0‘;
                else if (matrix[i][j] == ‘1‘) {
                    dp[i][j] = min(dp[i - 1][j - 1], min(dp[i][j - 1], dp[i - 1][j])) + 1;
                }
                res = max(res, dp[i][j]);
            }
        }
        return res * res;
    }
};

四.  Maximal Rectangle  Histogram

https://www.cnblogs.com/grandyang/p/4322653.html

基于pointer的解法,依次找到局部最大值分别进行处理,处理方法为分别求出以局部最大值为右边,所有左边分别能构成的最大面积。  最终通过一个全局最大值返回最优解。

class Solution {
public:
    int largestRectangleArea(vector<int> &height) {
        int res = 0;
        for (int i = 0; i < height.size(); ++i) {
            if (i + 1 < height.size() && height[i] <= height[i + 1]) {
                continue;
            }
            int minH = height[i];
            for (int j = i; j >= 0; --j) {
                minH = min(minH, height[j]);
                int area = minH * (i - j + 1);
                res = max(res, area);
            }
        }
        return res;
    }
};

基于单调递增栈的解法

在原始数组最右边加入一个0,单调递增依次进栈, 找到一个局部最大值,处理当前局部最大值前面所有比它小的数字。首先处理栈顶元素下标对应的高度,面积范围为以栈顶为高,范围为局部最大值为右边界(不包括), pop后的新的栈顶坐标为左边界(不包括)。

class Solution {
public:
    int largestRectangleArea(vector<int>& heights) {
        int res = 0;
        stack<int> st;
        heights.push_back(0);
        for (int i = 0; i < heights.size(); ++i) {
            while (!st.empty() && heights[st.top()] >= heights[i]) {
                int cur = st.top(); st.pop();
                res = max(res, heights[cur] * (st.empty() ? i : (i - st.top() - 1)));
            }
            st.push(i);
        }
        return res;
    }
};

五:Maximal Reactangle

类似于maximal square, 可以先构造一个新的二维数组,而后根据maximal histogram进行统计。

class Solution {
public:
    int maximalRectangle(vector<vector<char>>& matrix) {
        int m = matrix.size();
        if (m<1) return 0;
        int n = matrix[0].size();
        if (n<1) return 0;
        vector<vector<int>> matrix_int(m, vector<int>(n, 0));
        for (int i = 0; i < m; i++)
          for (int j = 0; j < n; j++) {
            matrix_int[i][j] = matrix[i][j] - ‘0‘;
            if (matrix_int[i][j] == 1) {
            if (i>0){
            matrix_int[i][j] += matrix_int[i-1][j];
            }
            }
        }
       int max_re = INT_MIN;
       for (int i=0; i < m; i++) {
          int max_tmp = maxHist(matrix_int[i]);
          if (max_re < max_tmp) max_re = max_tmp;
          cout << max_tmp << endl;
       }

      return max_re;
    }

private:
   int maxHist(vector<int> nums) {
       int len = nums.size();
       nums.push_back(0);
       int max_area = 0;
       for (int i=0; i < len; i++){
           if(nums[i] > nums[i+1]) {
               int max_local = 0;
               int min_h = nums[i];
               if (i==0) max_local = nums[i];
               else {
                  for(int j = i; j >=0; j--) {
                    if (nums[j] < min_h){
                        min_h = nums[j];
                    }
                    max_local = max(max_local, min_h * (i-j + 1));
                   }
               }
               if(max_local > max_area) max_area = max_local;
           }
      }
      return max_area;
   }
};

六.  WordBreak

DP[i] 表示 s[0,1,2 ... i-1] 是否能wordbreak成功,dp[i] = dp[j] && substr[j, j+1, .... i-1]为在word dict中的某个字符串。其中  0<=j <= i-1, 能找到一个就为true。C++ string.str的用法,第一个为开始字符下标,第二个参数为字符串长度。

class Solution {
public:
    bool wordBreak(string s, vector<string>& wordDict) {
        int len = s.size();
        if (len < 1) return false;
        set<string> dict;
        vector<bool> flags(len+1, false);
        flags[0] = true;
        for (int i=0; i < wordDict.size(); i++)
            dict.insert(wordDict[i]);
        for (int i = 1; i<= len; i++) {
            for (int j=0; j<i; j++) {
               if(flags[j] == true && dict.count(s.substr(j, i - j)) > 0){
                 flags[i] = true;
                 break;
               }
            }
        }
        return flags[len];
    }
};

原文地址:https://www.cnblogs.com/cookcoder-mr/p/11080050.html

时间: 2024-10-25 09:41:56

算法整理-动态规划和Two Pointers的相关文章

常见数据结构与算法整理总结(下)

这篇文章是常见数据结构与算法整理总结的下篇,上一篇主要是对常见的数据结构进行集中总结,这篇主要是总结一些常见的算法相关内容,文章中如有错误,欢迎指出. 一.概述 二.查找算法 三.排序算法 四.其它算法 五.常见算法题 六.总结 一.概述 以前看到这样一句话,语言只是工具,算法才是程序设计的灵魂.的确,算法在计算机科学中的地位真的很重要,在很多大公司的笔试面试中,算法掌握程度的考察都占据了很大一部分.不管是为了面试还是自身编程能力的提升,花时间去研究常见的算法还是很有必要的.下面是自己对于算法这

算法导论--动态规划(装配线调度)

装配线问题: 某个工厂生产一种产品,有两种装配线选择,每条装配线都有n个装配站.可以单独用,装配线1或2加工生产,也可以使用装配线i的第j个装配站后,进入另一个装配线的第j+1个装配站继续生产.现想找出通过工厂装配线的最快方法. 装配线i的第j个装配站表示为Si,j,在该站的装配时间是ai,j 如果从 Si,j装配站生产后,转移到另一个生产线继续生产所耗费的时间为ti,j 进入装配线花费时间ei,完成生产后离开装配线所耗费时间为xi 令f*表示通过生产所有路线中的最快的时间 令fi[j]表示从入

最短路径算法整理(二)

本文是最短路径算法整理的第二篇,想阅读第一篇的朋友能够点击下面链接: http://blog.csdn.net/hjd_love_zzt/article/details/26739593 这一篇博客继续以一些OJ上的题目为载体,整理一下最短路径算法.会陆续的更新... 1.HDU 2544 题目与分析:这道题抽象一下,还是:"求a到b的最短路径"..所须要的基本条件是:点数.边数.起点.终点 下面给出floyd.dijkstra.bellmanford三种最短路径算法关于这道题的解法:

算法整理(三):插入排序

插入排序很简单,就像打扑克.手里有个牌4,再来一张牌5就本能的放到第一个牌的右边.如果来了个3就从右往左扫描,只要左边的比这个待插入数字大就交换. 插入排序是一种稳定的排序方法,时间复杂度O(n*n),空间复杂度O(1),最好的情况下时间复杂度为O(1).即本来就是一个有序或者相等的数组,则只需比较n-1次即可.下为源码,只需三行代码即可. //============================================================================

常见数据结构与算法整理总结(上)

数据结构是以某种形式将数据组织在一起的集合,它不仅存储数据,还支持访问和处理数据的操作.算法是为求解一个问题需要遵循的.被清楚指定的简单指令的集合.下面是自己整理的常用数据结构与算法相关内容,如有错误,欢迎指出. 为了便于描述,文中涉及到的代码部分都是用Java语言编写的,其实Java本身对常见的几种数据结构,线性表.栈.队列等都提供了较好的实现,就是我们经常用到的Java集合框架,有需要的可以阅读这篇文章.Java - 集合框架完全解析 一.线性表 1.数组实现 2.链表 二.栈与队列 三.树

算法导论--动态规划(钢条切割)

钢条切割问题 现有一段长度为n英寸的钢条和一个价格表pi,求切割方案使销售利益最大rn最大 长度为n英寸的钢条共有2n?1种不同的切割方案,因为可以每个整英寸的位置都可以决定切割或者不切割. 为了得到rn最大,可以把这个问题分成子问题求解,先切一刀,再考虑余下的部分的最大收益即求 rn=max{pk+rn?k}(k=1,2,3-n-1), pk部分不进行继续切割,直接作为一个整体售出 ; rn?k部分继续切割,考虑所有的情况,分成子问题. 求出所有k值对应的收益最大者作为rn 也有可能不进行任何

算法整理(四):浅析快速排序的优化问题

前文介绍了快速排序的单边扫描和双边扫描,但么有做对比,今天来简单分析下. 一.单边扫描的缺点 单边扫描最大的缺点是每次都要交换,如果一个数组是 5 4 3 2 1,用单边扫描的话,则从4开始,4要和4交换一次,3要和3交换一次,依次类推,这种无意义的操作.正因此用双边扫描会更好,第一趟只需交换一次,就能得到1 4 3 2 5这样的数组.但双边扫描也是可以进一步优化的. 二.双边扫描的优化 优化一:对key值得选取应该使用随机选取的原则,而非第一个数字.意义大家都懂得. 优化二:前文的方法是挖坑法

经典算法宝典——动态规划思想(六)(2)

1.01背包问题 有N件物品和一个容量为V的背包,第i件物品的体积是c[i],价值是w[i].求解将哪些物品装入背包可使价值总和最大. 解析: 这是最基础的背包问题,特点是每种物品仅有一件,可以选择放或不放.用子问题定义状态,即f[i][v]表示前i件物品恰放入一个容量为v的背包可以获得的最大价值.其状态转移方程便是f[i][v] = max{f[i-1][v], f[i-1][v-c[i]]+w[i]},这个方程非常重要,基本上所有跟背包相关的问题的方程都是由它衍生出来的,所以有必要将它详细解

贪心算法和动态规划算法

动态规划和贪心算法都是一种递推算法 即均由局部最优解来推导全局最优解 (不从整体最优解出发来考虑,总是做出在当前看来最好的选择.) 不同点: 贪心算法 与动态规划的区别:贪心算法中,作出的每步贪心决策都无法改变,由上一步的最优解推导下一步的最优解,所以上一部之前的最优解则不作保留. 能使用贪心法求解的条件:是否能找出一个贪心标准.我们看一个找币的例子,如果一个货币系统有三种币值,面值分别为一角.五分和一分,求最小找币数时,可以用贪心法求解:如果将这三种币值改为一角一分.五分和一分,就不能使用贪心