最大子序和的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
}

 

转载于:https://www.cnblogs.com/TimLiuDream/p/10009791.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,下面是一个简单的电子通讯录程序的实现示例: ```go package main import ( "fmt" ) type Contact struct { name string number string email string } func main() { contacts := make(map[string]Contact) // 添加联系人 contacts["张三"] = Contact{name: "张三", number: "123456789", email: "zhangsan@example.com"} contacts["李四"] = Contact{name: "李四", number: "987654321", email: "lisi@example.com"} contacts["王五"] = Contact{name: "王五", number: "555555555", email: "wangwu@example.com"} // 显示联系人列表 fmt.Println("联系人列表:") for _, contact := range contacts { fmt.Printf("%s, %s, %s\n", contact.name, contact.number, contact.email) } // 根据名字查找联系人 fmt.Println("查找联系人:") name := "张三" if contact, ok := contacts[name]; ok { fmt.Printf("%s, %s, %s\n", contact.name, contact.number, contact.email) } else { fmt.Printf("找不到名字为 %s 的联系人\n", name) } // 修改联系人信息 fmt.Println("修改联系人:") name = "李四" if contact, ok := contacts[name]; ok { contact.number = "888888888" contact.email = "lisi@example.com" contacts[name] = contact fmt.Printf("%s, %s, %s\n", contact.name, contact.number, contact.email) } else { fmt.Printf("找不到名字为 %s 的联系人\n", name) } // 删除联系人 fmt.Println("删除联系人:") name = "王五" if _, ok := contacts[name]; ok { delete(contacts, name) fmt.Printf("已删除名字为 %s 的联系人\n", name) } else { fmt.Printf("找不到名字为 %s 的联系人\n", name) } // 显示修改后的联系人列表 fmt.Println("修改后的联系人列表:") for _, contact := range contacts { fmt.Printf("%s, %s, %s\n", contact.name, contact.number, contact.email) } } ``` 这个程序使用了一个 `Contact` 结构体来表示联系人,使用了一个 `map` 来存储所有联系人。程序中包含了添加、查找、修改、删除联系人等功能。你可以根据自己的需求来修改和扩展这个程序。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值