【LeetCode每日一题合集】2023.7.10-2023.7.16(dfs 换根DP)

news2024/11/24 12:46:09

文章目录

  • 16. 最接近的三数之和
    • 排序 + 双指针
  • 1911. 最大子序列交替和
    • 解法——动态规划
  • 2544. 交替数字和(简单模拟)
  • 931. 下降路径最小和(线性DP)
  • 979. 在二叉树中分配硬币⭐⭐⭐⭐⭐(dfs)
    • 算法分析
    • 补充:贡献法相关题目练习
      • 891. 子序列宽度之和
  • 18. 四数之和(排序 + 双指针 + 去重 + long)
  • 834. 树中距离之和⭐⭐⭐⭐⭐(两次 dfs)
    • 思路——冷静分析,列出式子
    • 算法分析⭐⭐⭐⭐⭐
    • 补充:相关题目列表(换根DP)🐂!

16. 最接近的三数之和

16. 最接近的三数之和
在这里插入图片描述

排序 + 双指针

class Solution {
    public int threeSumClosest(int[] nums, int target) {
        int n = nums.length, mn = Integer.MAX_VALUE, ans = 0;
        Arrays.sort(nums);
        for (int i = 0; i < n - 2; ++i) {
            for (int l = i + 1, r = n - 1; l < r; ) {
                int sum = nums[i] + nums[l] + nums[r];
                int diff = Math.abs(sum - target);
                if (diff < mn) {
                    mn = diff;
                    ans = sum;
                }

                if (sum < target) ++l;
                else if (sum > target) --r;
                else return ans;
            }
        }
        return ans;
    }
}

时间复杂度控制在 O ( n 2 ) O(n^2) O(n2)

1911. 最大子序列交替和

1911. 最大子序列交替和

在这里插入图片描述

解法——动态规划

  1. dp[i][0]和dp[i][1]分别表示以i为结果做加法和做减法的最大值
  2. 递推公式见代码
  3. dp数组初始化dp[0]即可
  4. 从前向后遍历
class Solution {
    public long maxAlternatingSum(int[] nums) {
        int n = nums.length;
        long[][] dp = new long[n][2];
        dp[0][0] = nums[0];
        for (int i = 1; i < n; ++i) {
            dp[i][0] = Math.max(dp[i - 1][0], dp[i - 1][1] + nums[i]);
            dp[i][1] = Math.max(dp[i - 1][1], dp[i - 1][0] - nums[i]);
        }
        return Math.max(dp[n - 1][0], dp[n - 1][1]);
    }
}

这题很类似 122. 买卖股票的最佳时机 II
,只不过可以在没有股票的情况下先卖一个。
这种思想写出来的代码如下:

class Solution {
    public long maxAlternatingSum(int[] prices) {
        int n = prices.length;
        long sell = prices[0], buy = 0;
        for (int i = 1; i < n; ++i) {
            long newSell = Math.max(sell, buy + prices[i]);
            buy = Math.max(buy, sell - prices[i]);
            sell = newSell;
        }
        return sell;
    }
}

2544. 交替数字和(简单模拟)

https://leetcode.cn/problems/alternating-digit-sum/
在这里插入图片描述

简单模拟即可。

 class Solution {
    public int alternateDigitSum(int n) {
        int sign = 1, ans = 0;
        while (n != 0) {
            ans += sign * (n % 10);
            n /= 10;
            sign *= -1;
        }
        return -sign * ans;
    }
}

931. 下降路径最小和(线性DP)

https://leetcode.cn/problems/minimum-falling-path-sum/
在这里插入图片描述

class Solution {
    public int minFallingPathSum(int[][] matrix) {
        int m = matrix.length, n = matrix[0].length;
        int[][] dp = new int[m][n];
        dp[0] = Arrays.copyOf(matrix[0], n);
        for (int i = 1; i < m; ++i) {
            for (int j = 0; j < n; ++j) {
                dp[i][j] = dp[i - 1][j];
                if (j - 1 >= 0) dp[i][j] = Math.min(dp[i][j], dp[i - 1][j - 1]);
                if (j + 1 < m) dp[i][j] = Math.min(dp[i][j], dp[i - 1][j + 1]);
                dp[i][j] += matrix[i][j];
            }
        }
        return Arrays.stream(dp[m - 1]).min().getAsInt();
    }
}

979. 在二叉树中分配硬币⭐⭐⭐⭐⭐(dfs)

https://leetcode.cn/problems/distribute-coins-in-binary-tree/
在这里插入图片描述

提示:
1<= N <= 100
0 <= node.val <= N

没有靠自己第一眼就做出来!

dfs (root) 的返回值表示 root 的父节点需要从 root 节点拿走几个金币才能让 root 处的金币数量为 1。

https://leetcode.cn/problems/distribute-coins-in-binary-tree/solutions/2339545/zai-er-cha-shu-zhong-fen-pei-ying-bi-by-e4poq/

class Solution {
    int ans = 0;

    public int distributeCoins(TreeNode root) {
        dfs(root);
        return ans;
    }

    // 返回值是root的父节点需要从root拿走几个金币才能让root是1
    public int dfs(TreeNode root) {
        int moveLeft = 0, moveRight = 0;
        if (root == null) return 0;
        if (root.left != null) moveLeft = dfs(root.left);
        if (root.right != null) moveRight = dfs(root.right);
        ans += Math.abs(moveLeft) + Math.abs(moveRight);
        return moveLeft + moveRight + root.val - 1;
    }
}

算法分析

在这里插入图片描述
路径是由边组成的,所有路径长度之和,等同于把「每条边出现在多少条路径中」相加。这种技巧叫做贡献法

补充:贡献法相关题目练习

更多题目可见:【算法】贡献法相关题目练习

891. 子序列宽度之和

https://leetcode.cn/problems/sum-of-subsequence-widths/
在这里插入图片描述

思路:
在这里插入图片描述

考虑每个元素会作为几个子序列的最小值,又会作为几个子序列的最大值。

class Solution {
    public int sumSubseqWidths(int[] nums) {
        final int MOD = (int)1e9 + 7;
        Arrays.sort(nums);          // 看到是序列,和顺序无关,可以排序!
        int n = nums.length;
        // 预处理出 2 的幂次
        int[] pow2 = new int[n];
        pow2[0] = 1;
        for (int i = 1; i < n; ++i) {
            pow2[i] = pow2[i - 1] * 2 % MOD;
        }
        long ans = 0;
        for (int i = 0; i < n; ++i) {
            // 计算 nums[i] 作为最大值的贡献减去作为最小值的贡献
            ans = (ans + (long)(pow2[i] - pow2[n - 1 - i]) * nums[i]) % MOD;
        }
        return (int)(ans + MOD) % MOD;
    }
}

18. 四数之和(排序 + 双指针 + 去重 + long)

18. 四数之和
在这里插入图片描述

排序 + 双指针 + 去重 + long

class Solution {
    public List<List<Integer>> fourSum(int[] nums, int target) {
        List<List<Integer>> ans = new ArrayList();
        Arrays.sort(nums);
        int n = nums.length;
        for (int i = 0; i < n - 3; ++i) {
            if (i > 0 && nums[i] == nums[i - 1]) continue;
            for (int j = i + 1; j < n - 2; ++j) {
                if (j > i + 1 && nums[j] == nums[j - 1]) continue; 
                long t = (long)target - nums[i] - nums[j];
                for (int l = j + 1, r = n - 1; l < r; ) {
                    if (l > j + 1 && nums[l] == nums[l - 1]) {
                        ++l;
                        continue;
                    }
                    long x = nums[l] + nums[r];
                    if (x < t) l++;
                    else if (x > t) r--;
                    else ans.add(List.of(nums[i], nums[j], nums[l++], nums[r--]));
                }
            }
        }
        return ans;
    }
}

834. 树中距离之和⭐⭐⭐⭐⭐(两次 dfs)

834. 树中距离之和

在这里插入图片描述

思路——冷静分析,列出式子

https://leetcode.cn/problems/sum-of-distances-in-tree/solutions/103325/c-liang-ci-dfsde-dao-da-an-by-congwang357-2/

将问题拆分:对于两个相邻节点A和B,将树拆分为两个子树,根节点分别为A和B,A节点到其他所有节点的距离和 ans(A) = A子树中所有节点到A节点的距离和sum(A) + B子树中所有节点到B节点的距离和sum(B) + B子树的大小cnt(B);
同理,ans(B) = sum(B) + sum(A) + cnt(A);

由此我们得到: ans(A) = sum(A) + sum(B) + cnt(B); ans(B) = sum(B) + sum(A) + cnt(A);
则,两个相邻接点的解之间的关系为:ans(A) = ans(B) - cnt(A) + cnt(B) = ans(B) - cnt(A) + (N - cnt(A));

因此,对于根节点root的任意子节点child,ans(child) = ans(root) - cnt(child) + N - cnt(child);

class Solution {
    List<Integer>[] g;
    int[] ans, size;    // size是各个节点作为根节点的子树大小
    int n;
    public int[] sumOfDistancesInTree(int n, int[][] edges) {
        g = new ArrayList[n];
        Arrays.setAll(g, e -> new ArrayList<Integer>());
        for (int[] edge: edges) {
            int x = edge[0], y = edge[1];
            g[x].add(y);
            g[y].add(x);
        }
        this.n = n;
        ans = new int[n];
        size = new int[n];
        Arrays.fill(size, 1);

        dfs(0, -1, 0);
        reroot(0, -1);
        return ans;
    }

    // 求ans[0]和各个size[i]
    void dfs(int x, int fa, int depth) {
        ans[0] += depth;                    // depth 是 0 到 x 的距离
        for (int y: g[x]) {
            if (y != fa) {
                dfs(y, x, depth + 1);
                size[x] += size[y];         // 累加 x 的儿子 y 的子树大小
            }
        }
    }

    // 求答案
    void reroot(int x, int fa) {
        for (int y: g[x]) {
            if (y != fa) {
                ans[y] = ans[x] + n - 2 * size[y];
                reroot(y, x);
            }
        }
    }
}

算法分析⭐⭐⭐⭐⭐

https://leetcode.cn/problems/sum-of-distances-in-tree/solutions/2345592/tu-jie-yi-zhang-tu-miao-dong-huan-gen-dp-6bgb/
在这里插入图片描述
我们得到了重要公式:
a n s [ y ] = a n s [ x ] + n − 2 ∗ s i z e [ y ] ans[y] = ans[x] + n - 2 * size[y] ans[y]=ans[x]+n2size[y]

如何理解?
y 和以 y为根的子树的距离相比 x 与 以 y为根的子树的距离 少了 cnt[y]
除了 以 y为根的子树,剩下的节点数量是 n - cnt[y],这些和 y 的距离相比 和 x 的距离多了 n - cnt[y]

因此:ans[y] = ans[x] + n - 2 * size[y]

补充:相关题目列表(换根DP)🐂!

更多题目参见:【算法】换根DP

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/760938.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

PWM呼吸灯+流水灯设计

完成任务&#xff1a; 在流水灯基础上加入pwm呼吸灯设计&#xff0c;关于pwm呼吸灯设计可以看博主上一篇博客PWM呼吸灯设计 &#xff0c;开发板上灯每两秒进行一次切换&#xff0c;每一个的亮灭间隔为一秒。 代码参考&#xff1a; module pwm_led_change(input wire …

软件测试人员和程序开发人员是死对头吗?

这两天闲来无事刷知乎&#xff0c;看到有些朋友问到关于测试与开发的关系&#xff0c;在这里想和大家稍微来聊一聊这个事儿。 有的人说呢&#xff0c;测试和开发是死对头&#xff1b;也有人说测试和开发是处在对立面的&#xff1b;还有人说测试与开发两者都不能互相理解。当然&…

再战算法-奋进

再战算法-奋进 算法入门痛苦经历总结收获 算法入门 在大学期间我直至觉得【算法】是很重要的一项&#xff0c;最开始接触的是c语言&#xff0c;算是第一门接触的&#xff0c;给了我很大的惊喜&#x1f970;&#xff0c;大二下的时候开始接触到Java语言&#xff0c;通过Java的入…

【C++进阶】C++11基础

文章目录 一、C11简介二、统一的列表初始化1. &#xff5b;&#xff5d;初始化2、std::initializer_list 三、 声明1.auto2. decltype3.nullptr 三、范围for 一、C11简介 在2003年C标准委员会曾经提交了一份技术勘误表(简称TC1)&#xff0c;使得C03这个名字已经取代了C98称为C…

Python3实现画小提琴图(包含分组)

说在前面 Python如何画一个小提琴图呢?先看下必备的数据集合(自己构建,样式参考) 默认必有X列、Y列(数值),画分组需要包含分组的列group等数据参数准备 可以参考下面的数据样例: 除此之外,对于画图使用的参数,提前准备的知识如下: sns.violinplot所必备的参数…

数据结构-堆排序代码实现(详解)

内容&#xff1a;堆排序的代码实现及注解&#xff0c;思路详解 代码实现&#xff1a; 交换函数&#xff1a; void Swap(int* p1, int* p2) {int tmp *p1;*p1 *p2;*p2 tmp; } 堆排序函数&#xff1a; 1 向下调整建堆函数&#xff1a;这里建立大堆&#xff0c;小堆思路也…

Litedram仿真验证(三):AXI接口完成仿真(FPGA/Modelsim)

日常唠嗑 不知不觉&#xff0c;从开始接触Litedram已经过去了4个月&#xff0c;期间断断续续做了好多其他任务&#xff0c;导致进度比较慢&#xff0c;直到前天才把Litedram完全仿真起来。&#xff08;坑很多&#xff0c;很多东西需要注意&#xff09; 目录 日常唠嗑一、AXI用…

C—数据的储存(下)

文章目录 前言&#x1f31f;一、练习一下&#x1f30f;1.例一&#x1f30f;2.例二&#x1f30f;3.例三&#x1f30f;4.例四 &#x1f31f;二、浮点型在内存中的储存&#x1f30f;1.浮点数&#x1f30f;2.浮点数存储&#x1f4ab;&#xff08;1&#xff09;.二进制浮点数&#x…

快速入门java微服务架构SpringBoot之一

Springboot概念&#xff1a; Springboot提供了一种快速使用Springboot的方式&#xff0c;基于约定优于配置的思想。 可以让开发人员不必在配置与逻辑业务之间进行思维的切换&#xff0c;全身心的投入到逻辑业务的代码编辑中&#xff0c;从而大大提高了开发的效率&#xff0c;…

matlab主成分分析算法在人脸识别的具体应用

主成分析&#xff08;Principal Component Analysis&#xff0c;简称PCA&#xff09;是一种常用的降维算法&#xff0c;可以将高维数据转化为低维数据&#xff0c;同时保留原始数据的最重要特征。PCA算法在人脸识别中有广泛的应用&#xff0c;可以提取人脸图像中的主要特征&…

《啊哈算法》第二章--队列 栈 链表

文章目录 前言一、数据结构基础知识(衔接知识)二、队列三、栈四、链表总结 前言 上一节我们学习了排序算法当中的快速排序 冒泡排序 桶排序 &#xff0c;那么本节得主要学习内容是队列 栈 链表得相关数据结构得知识 一、数据结构基础知识(衔接知识) 基于学习这本书得都是一些…

《英雄联盟》丢失d3dcompiler_47.dll怎么办,推荐这个修复方案

不知道大家有么有遇到过&#xff0c;在打开《英雄联盟》的时候&#xff0c;计算机提示丢失d3dcompiler_47.dll&#xff0c;无法继续执行此代码。d3dcompiler_47.dll是一个动态链接库文件&#xff0c;它是与Direct3D编译器相关的组件之一。像是photoshop等软件&#xff0c;英雄联…

IEEE754 标准是如何制定浮点数的存储的

1. IEEE754 标准简介 IEEE754 标准是一种用于浮点数表示和运算的标准&#xff0c;由国际电工委员会&#xff08;IEEE&#xff09;制定。它定义了浮点数的编码格式、舍入规则以及基本的算术运算规则&#xff0c;旨在提供一种可移植性和一致性的方式来表示和处理浮点数 IEEE754 …

c#使用ThreadPool

说到ThreadPool&#xff0c;都知道是线程池。在c#中&#xff0c;有很多方式可以实现线程。从时间上来排序&#xff0c;大概是这样的&#xff0c;Thread&#xff0c;backgroundworker&#xff0c;ThreadPool&#xff0c;Parallel&#xff0c;Task。其中后面2种是最新的&#xff…

第十八章:Auto-DeepLab:用于语义图像分割的层次化神经架构搜索

0.摘要 最近&#xff0c;神经架构搜索&#xff08;NAS&#xff09;已经成功地识别出在大规模图像分类任务上超越人工设计的神经网络架构。在本文中&#xff0c;我们研究了NAS在语义图像分割任务中的应用。现有的工作通常集中在搜索可重复的基本单元结构&#xff0c;而手动设计控…

一些有趣的Git学习资料

Git 可以说是程序员必备的技能之一了&#xff0c;基于 Github/Gitlab 以及相关工作流的使用&#xff0c;Git 已经融入到了我们的日常工作中&#xff0c;这里分享一些有趣的 Git 学习资料&#xff0c;希望可以帮助大家更好的理解 Git。 1. Welcome to Learn Git Branching 以动…

第二周笔记

public class Calc { //加法, 把和作为一个结果返回出去, 返回给调用者 public int add3(int num1, int num2){ if(num1 0 && num2 0){ return 0; //隐式包含一个if-else结构 } //使用return 关键字 return nu…

【Linux操作系统】多线程抢票逻辑——学习互斥量(锁)函数接口

文章目录 1.进程线程间的互斥相关背景概念2.联系代码学习同步互斥问题3.互斥量&#xff08;锁&#xff09;的函数接口3.1初始化互斥量3.2销毁互斥量3.3互斥量加锁和解锁3.4改进多线程抢票代码 1.进程线程间的互斥相关背景概念 临界资源&#xff1a;多线程执行流共享的资源就叫…

beego验证码(配置到redis存储)

我们定义一个全局变量用于存储redis连接 RedisDb *redis.Client 然后连接 redis 这一块我们将redis信息写到app.conf文件里了&#xff1a; redisDb 1 redisAddr "127.0.0.1:6379" redisPwd "" package initializeimport ("beego_learning/global&q…

【Java基础教程】(十四)面向对象篇 · 第八讲:多态性详解——向上及向下转型、关键字 final与 instanceof的作用~

Java基础教程之面向对象 第八讲 本节学习目标1️⃣ final 关键字1.1 final类1.2 final方法1.3 final属性 2️⃣ 多态性2.1 向上转型2.2 向下转型2.3 关键字 instanceof &#x1f33e; 总结 本节学习目标 掌握final 关键字的主要作用及使用&#xff1b;掌握对象多态性的概念以…