zoukankan      html  css  js  c++  java
  • 最大子序和的golang实现

    给定一个整数数组 nums ,找到一个具有最大和的连续子数组(子数组最少包含一个元素),返回其最大和。

    输入: [-2,1,-3,4,-1,2,1,-5,4],
    输出: 6
    解释: 连续子数组 [4,-1,2,1] 的和最大,为 6
    首先理解题意:

    题目讲究的是连续,那我们可以假设一下,我们现在以下标为0的值为基准值,那么下一位就分为两种情况了:

    • 下一位为负数,那么我们就要与基准值比较,看哪个大
    • 下一位为正数,那么我们肯定是要相加了
    核心代码:
    //取出第一位为基准值
        sum := nums[0]
        res := sum
        //下标从1开始
        for i := 1; i < len(nums); i++ {
            //当值是负数的时候,就要与res比较,取出最大的
            //当值为正数的时候,就要与res相加,那么才会更大
            if sum < 0 {
                sum = nums[i]
            } else {
                sum += nums[i]
            }
            res = max(res, sum)
        }

    整体代码:

    package main
    
    import (
        "fmt"
    )
    
    func maxSubArray(nums []int) int {
        //取出第一位为基准值
        sum := nums[0]
        res := sum
        //下标从1开始
        for i := 1; i < len(nums); i++ {
            //当值是负数的时候,就要与res比较,取出最大的
            //当值为正数的时候,就要与res相加,那么才会更大
            if sum < 0 {
                sum = nums[i]
            } else {
                sum += nums[i]
            }
            res = max(res, sum)
        }
    
        return res
    }
    
    func max(a, b int) int {
        if a > b {
            return a
        }
        return b
    }
    
    func main() {
        nums := []int{-2, 1, -3, 4, -1, 2, 1, -5, 4}
        //nums := []int{-2, 1}
        fmt.Println(maxSubArray(nums))
    }

    经查找上方的算法叫扫描法,时间复杂度为O(n)。

    下面我们尝试一下使用动态规划法来解题:
    解题思路:

    设sum[i]为以第i个元素结尾且和最大的连续子数组。假设对于元素i,所有以它前面的元素结尾的子数组的长度都已经求得,那么以第i个元素结尾且和最大的连续子数组实际上,要么是以第i-1个元素结尾且和最大的连续子数组加上这个元素,要么是只包含第i个元素,即sum[i] = max(sum[i-1] + a[i], a[i])。可以通过判断sum[i-1] + a[i]是否大于a[i]来做选择,而这实际上等价于判断sum[i-1]是否大于0。由于每次运算只需要前一次的结果,因此并不需要像普通的动态规划那样保留之前所有的计算结果,只需要保留上一次的即可,因此算法的时间和空间复杂度都很小

    完整代码:
    //动态规划解法
    //假设sum[i]为以第i个元素结尾且和最大的连续子数组。
    //假设对于元素i,所有以它前面的元素结尾的子数组的长度都已经求得
    //那么以第i个元素结尾且和最大的连续子数组实际上,要么是以第i-1个元素结尾且和最大的连续子数组加上这个元素,要么是只包含第i个元素
    //即sum[i] = max(sum[j:i-1] + a[i], a[i])。//j是数组的某个下标,0<=j<i-1<n
    //可以通过判断sum[i-1] + a[i]是否大于a[i]来做选择,而这实际上等价于判断sum[i-1]是否大于0。
    //由于每次运算只需要前一次的结果,因此并不需要像普通的动态规划那样保留之前所有的计算结果,只需要保留上一次的即可,因此算法的时间和空间复杂度都很小
    func maxSubArrayKMP(nums []int) int {
        res, sum := nums[0], nums[0]
        for i := 0; i < len(nums); i++ {
            //这里是核心判断
            //当前几项大于0,那就相加
            if (res > 0) {
                res += nums[i]
            } else {//如果不是,那就直接是当前项
                res = nums[i]
            }
            if (sum < res) {
                sum = res
            }
        }
        return res
    }
  • 相关阅读:
    《java入门第一季》之面向对象(static关键字)
    《java入门第一季》之面向对象(面向对象案例详解)
    《java入门第一季》之面向对象面试题(面向对象都做了哪些事情)
    《java入门第一季》之面向对象(成员方法)
    《android入门第一季》之android目录结构详解
    Vue 中的 Props 与 Data 细微差别,你知道吗?
    使用Vue 3.0做JSX(TSX)风格的组件开发
    vue中Axios的封装和API接口的管理
    在 Vue.js 中制作自定义选择组件
    webpack打包原理
  • 原文地址:https://www.cnblogs.com/TimLiuDream/p/10009791.html
Copyright © 2011-2022 走看看