力扣494 目标和 java实现

494.目标和

给你一个非负整数数组 nums 和一个整数 target

向数组中的每个整数前添加 '+''-' ,然后串联起所有整数,可以构造一个 表达式

  • 例如,nums = [2, 1] ,可以在 2 之前添加 '+' ,在 1 之前添加 '-' ,然后串联起来得到表达式 "+2-1"

返回可以通过上述方法构造的、运算结果等于 target 的不同 表达式 的数目。

示例 1:

复制代码
输入:nums = [1,1,1,1,1], target = 3
输出:5
解释:一共有 5 种方法让最终目标和为 3 。
-1 + 1 + 1 + 1 + 1 = 3
+1 - 1 + 1 + 1 + 1 = 3
+1 + 1 - 1 + 1 + 1 = 3
+1 + 1 + 1 - 1 + 1 = 3
+1 + 1 + 1 + 1 - 1 = 3

示例 2:

复制代码
输入:nums = [1], target = 1
输出:1

提示:

  • 1 <= nums.length <= 20
  • 0 <= nums[i] <= 1000
  • 0 <= sum(nums[i]) <= 1000
  • -1000 <= target <= 1000

该题我写了两种方法,一种是回溯法,本题也属于一种选或不选的问题,也能使用回溯法,就是时间复杂度要高得多,另一种是动态规划,时间复杂度则相对较低。

首先是回溯法,只需要递归考虑选或不选,最后判断是否和目标数相同即可。

java 复制代码
public class hot {
    public static void main(String[] args) {  // 测试用
        int[] nums = {0};
        hot hot = new hot();
        System.out.println(hot.findTargetSumWays(nums, 0));
    }


    int res = 0;

    public  int findTargetSumWays(int[] nums, int target) {
        boolean[] isOrNot = new boolean[nums.length];
        helper(nums, target,0, 0);

        return res;
    }

    public void helper(int[] nums, int target, int cur, int sum){
        if (cur >= nums.length){
            if (sum == target){
                res++;
            }
            return;
        }
        helper(nums, target, cur + 1, sum + nums[cur]);
        helper(nums, target, cur + 1, sum - nums[cur]);
    }
}

动态规划法比较复杂,要搞清楚动态规划数组的含义才容易理解。dp[i][j]:使用 下标为[0, i]的nums[i]能够凑满j(包括j)这么大容量的包,有dp[i][j]种方法。dp[i][j]处的结果由dp[i - 1][j] 和 dp[i - 1][j - nums[i]]确定,若num[i] <= j时则为两者相加,否则则为dp[i - 1][j]

java 复制代码
public class hot {
    public static void main(String[] args) {  // 测试用
        int[] nums = {0};
        hot hot = new hot();
        System.out.println(hot.findTargetSumWays(nums, 0));
    }


    public  int findTargetSumWays(int[] nums, int target) {  
        int sum = 0;
        for (int i = 0; i < nums.length; i++) {
            sum +=  nums[i];
        }
        if ((sum + target) % 2 == 1 || Math.abs(target) > sum){
            return 0;
        }
        int size = (sum + target) / 2;
        int[][] dp = new int[nums.length][size + 1];
        dp[0][0] = 1;
        if (nums[0] <= size){
            dp[0][nums[0]] = 1;
            if (nums[0] == 0){
                dp[0][0] = 2;
            }
        }
        for (int i = 1; i < nums.length; i++) {
            for (int j = 0; j < size + 1; j++) {
               if (nums[i] <= j){
                    dp[i][j] = dp[i - 1][j] + dp[i - 1][j - nums[i]];
                }else {
                    dp[i][j] = dp[i - 1][j];
                }
            }
        }
        return dp[nums.length - 1][size];
    }
}

以上为记录分享用,代码较差请见谅。

相关推荐
草履虫建模15 小时前
力扣算法 1768. 交替合并字符串
java·开发语言·算法·leetcode·职场和发展·idea·基础
naruto_lnq17 小时前
分布式系统安全通信
开发语言·c++·算法
Jasmine_llq17 小时前
《P3157 [CQOI2011] 动态逆序对》
算法·cdq 分治·动态问题静态化+双向偏序统计·树状数组(高效统计元素大小关系·排序算法(预处理偏序和时间戳)·前缀和(合并单个贡献为总逆序对·动态问题静态化
qq_2975746717 小时前
【实战教程】SpringBoot 实现多文件批量下载并打包为 ZIP 压缩包
java·spring boot·后端
老毛肚17 小时前
MyBatis插件原理及Spring集成
java·spring·mybatis
学嵌入式的小杨同学18 小时前
【Linux 封神之路】信号编程全解析:从信号基础到 MP3 播放器实战(含核心 API 与避坑指南)
java·linux·c语言·开发语言·vscode·vim·ux
lang2015092818 小时前
JSR-340 :高性能Web开发新标准
java·前端·servlet
Re.不晚18 小时前
Java入门17——异常
java·开发语言
爱吃rabbit的mq18 小时前
第09章:随机森林:集成学习的威力
算法·随机森林·集成学习
缘空如是18 小时前
基础工具包之JSON 工厂类
java·json·json切换