前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >2021-06-28:最接近目标值的子序列和。给你一个整数数组 nums 和一个目标值 goal 。你需要从 nums 中选出一

2021-06-28:最接近目标值的子序列和。给你一个整数数组 nums 和一个目标值 goal 。你需要从 nums 中选出一

作者头像
福大大架构师每日一题
发布2021-08-05 16:13:36
5120
发布2021-08-05 16:13:36
举报

2021-06-28:最接近目标值的子序列和。给你一个整数数组 nums 和一个目标值 goal 。你需要从 nums 中选出一个子序列,使子序列元素总和最接近 goal 。也就是说,如果子序列元素和为 sum ,你需要 最小化绝对差 abs(sum - goal) 。返回 abs(sum - goal) 可能的 最小值 。注意,数组的子序列是通过移除原始数组中的某些元素(可能全部或无)而形成的数组。输入:nums = [7,-9,15,-2], goal = -5。输出:1。解释:选出子序列 [7,-9,-2] ,元素和为 -4 。绝对差为 abs(-4 - (-5)) = abs(1) = 1 ,是可能的最小值。

示例 1:

输入:nums = [5,-7,3,5], goal = 6

输出:0

解释:选择整个数组作为选出的子序列,元素和为 6 。

子序列和与目标值相等,所以绝对差为 0 。

示例 2:

输入:nums = [7,-9,15,-2], goal = -5

输出:1

解释:选出子序列 [7,-9,-2] ,元素和为 -4 。

绝对差为 abs(-4 - (-5)) = abs(1) = 1 ,是可能的最小值。

示例 3:

输入:nums = [1,2,3], goal = -7

输出:7

来源:力扣(LeetCode)

链接:https://leetcode-cn.com/problems/closest-subsequence-sum

著作权归领扣网络所有。商业转载请联系官方授权,非商业转载请注明出处。

福大大 答案2021-06-28:

本题数据量描述:

1 <= nums.length <= 40,

-10^7 <= nums[i] <= 10^7,

-10^9 <= goal <= 10^9,

通过这个数据量描述可知,需要用到分治,因为数组长度不大,

而值很大,用动态规划的话,表会爆。

时间复杂度是O(2的20次方*20),空间复杂度是O(2的20次方*2)。

代码用golang编写。代码如下:

代码语言:javascript
复制
package main

import (
    "fmt"
    "sort"
)

func main() {
    arr := []int{1, 3, 5, 7, 9, 2, 4, 6, 8, 19}
    ret := minAbsDifference(arr, 11)
    fmt.Println(ret)
}

var l []int = make([]int, 1<<20)
var r []int = make([]int, 1<<20)

func minAbsDifference(nums []int, goal int) int {
    if len(nums) == 0 {
        return goal
    }
    le := process(nums, 0, len(nums)>>1, 0, 0, l)
    re := process(nums, len(nums)>>1, len(nums), 0, 0, r)
    sort.Ints(l[0 : le+1])
    sort.Ints(r[0 : re+1])
    re--
    ans := getAbs(goal)
    for i := 0; i < le; i++ {
        rest := goal - l[i]
        for re > 0 && getAbs(rest-r[re-1]) <= getAbs(rest-r[re]) {
            re--
        }
        ans = getMin(ans, getAbs(rest-r[re]))
    }
    return ans
}

func process(nums []int, index int, end int, sum int, fill int, arr []int) int {
    if index == end {
        arr[fill] = sum
        fill++
    } else {
        fill = process(nums, index+1, end, sum, fill, arr)
        fill = process(nums, index+1, end, sum+nums[index], fill, arr)
    }
    return fill
}

func getMin(a int, b int) int {
    if a < b {
        return a
    } else {
        return b
    }
}
func getAbs(a int) int {
    if a < 0 {
        return -a
    } else {
        return a
    }
}

执行结果如下:

***

[左神java代码](https://github.com/algorithmzuo/coding-for-great-offer/blob/main/src/class03/Code06_ClosestSubsequenceSum.java)

本文参与 腾讯云自媒体分享计划,分享自微信公众号。
原始发表:2021-06-28,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 福大大架构师每日一题 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档