代码随想录-Day44

news2024/12/24 8:01:59

322. 零钱兑换

给你一个整数数组 coins ,表示不同面额的硬币;以及一个整数 amount ,表示总金额。

计算并返回可以凑成总金额所需的 最少的硬币个数 。如果没有任何一种硬币组合能组成总金额,返回 -1 。

你可以认为每种硬币的数量是无限的。

示例 1:

输入:coins = [1, 2, 5], amount = 11
输出:3
解释:11 = 5 + 5 + 1
示例 2:

输入:coins = [2], amount = 3
输出:-1
示例 3:

输入:coins = [1], amount = 0
输出:0
在这里插入图片描述

class Solution {
    public int coinChange(int[] coins, int amount) {
        int max = Integer.MAX_VALUE;
        int[] dp = new int[amount + 1];
        //初始化dp数组为最大值
        for (int j = 0; j < dp.length; j++) {
            dp[j] = max;
        }
        //当金额为0时需要的硬币数目为0
        dp[0] = 0;
        for (int i = 0; i < coins.length; i++) {
            //正序遍历:完全背包每个硬币可以选择多次
            for (int j = coins[i]; j <= amount; j++) {
                //只有dp[j-coins[i]]不是初始最大值时,该位才有选择的必要
                if (dp[j - coins[i]] != max) {
                    //选择硬币数目最小的情况
                    dp[j] = Math.min(dp[j], dp[j - coins[i]] + 1);
                }
            }
        }
        return dp[amount] == max ? -1 : dp[amount];
    }
}

这段Java代码实现了一个经典的动态规划问题——“完全背包问题”的一个应用场景:给定不同面额的硬币coins和一个总金额amount,计算最少需要多少个硬币凑出这个金额,如果不可能凑出则返回-1。这里是使用完全背包的思路,即每种硬币可以无限使用。

代码解析:

  1. 初始化:首先,定义一个dp数组,长度为amount + 1,并将其所有值初始化为Integer.MAX_VALUE,表示在没有计算之前,达到每个金额所需的最小硬币数为正无穷大。例外的是,dp[0]初始化为0,因为当金额为0时,不需要任何硬币。

  2. 双重循环

    • 外层循环遍历硬币数组coins,即遍历每一种硬币面额。
    • 内层循环从当前硬币的面额coins[i]开始遍历到总金额amount。这是因为在遍历到的金额j上,只有当j至少为当前硬币面额时,才有可能使用当前硬币去构成这个金额。
  3. 状态转移方程:对于内层循环中的每个j,如果dp[j - coins[i]]不是初始的最大值(即存在一种方式可以构成j - coins[i]的金额),则考虑使用一个面额为coins[i]的硬币,更新dp[j]dp[j]dp[j - coins[i]] + 1中的较小值。这里dp[j - coins[i]] + 1表示在构成j - coins[i]的基础上再加一个面额为coins[i]的硬币。

  4. 返回结果:最后,检查dp[amount]是否仍为Integer.MAX_VALUE,如果是,则说明没有找到任何组合可以凑成总金额,返回-1;否则返回dp[amount],即最少需要的硬币数。

总结:

这段代码通过动态规划的完全背包方法,有效解决了最少硬币数量问题,时间复杂度为O(coins.length * amount),空间复杂度为O(amount)。

279. 完全平方数

给你一个整数 n ,返回 和为 n 的完全平方数的最少数量 。

完全平方数 是一个整数,其值等于另一个整数的平方;换句话说,其值等于一个整数自乘的积。例如,1、4、9 和 16 都是完全平方数,而 3 和 11 不是。

示例 1:

输入:n = 12
输出:3
解释:12 = 4 + 4 + 4
示例 2:

输入:n = 13
输出:2
解释:13 = 4 + 9
在这里插入图片描述

方法一:

class Solution {
    // 版本一,先遍历物品, 再遍历背包
    public int numSquares(int n) {
        int max = Integer.MAX_VALUE;
        int[] dp = new int[n + 1];
        //初始化
        for (int j = 0; j <= n; j++) {
            dp[j] = max;
        }
	//如果不想要寫for-loop填充數組的話,也可以用JAVA內建的Arrays.fill()函數。
	//Arrays.fill(dp, Integer.MAX_VALUE);
	
        //当和为0时,组合的个数为0
        dp[0] = 0;
        // 遍历物品
        for (int i = 1; i * i <= n; i++) {
            // 遍历背包
            for (int j = i * i; j <= n; j++) {
                //if (dp[j - i * i] != max) {
                    dp[j] = Math.min(dp[j], dp[j - i * i] + 1);
                //}
		//不需要這個if statement,因爲在完全平方數這一題不會有"湊不成"的狀況發生( 一定可以用"1"來組成任何一個n),故comment掉這個if statement。
            }
        }
        return dp[n];
    }
}

这段Java代码是解决完全平方数问题的一个动态规划实现,目标是找出最少数量的完全平方数(例如1, 4, 9, 16…)之和,使其等于给定的正整数n。代码采用了“先遍历物品,再遍历背包”的动态规划策略,这里的“物品”指的是完全平方数,而“背包”则是我们需要达到的目标和n

代码解析

  1. 初始化dp数组:首先,创建一个长度为n+1的数组dp,其中dp[j]表示和为j时所需的最少完全平方数的个数。初始化所有dp[j]Integer.MAX_VALUE,表示初始时没有找到任何组合。但因为任何正整数都可以由无数个1^2组成,所以实际上不需要初始化为Integer.MAX_VALUE,直接初始化为一个较大的数即可,或明确知道最小组合数为1(当j > 0时)。注释中提到的Arrays.fill(dp, Integer.MAX_VALUE);是一种更简洁的初始化方式,但在这个特定问题上下文里,初始化为极大值然后在特定条件下更新的逻辑是多余的。

  2. 初始化dp[0]:当和为0时,不需要任何完全平方数,所以dp[0]设置为0。

  3. 双重循环

    • 外层循环遍历所有可能的完全平方数(由1^2√n的平方),用i表示当前考虑的完全平方数的根。
    • 内层循环遍历从当前完全平方数i*i开始到目标和n的所有可能和j。对于每个j,如果可以通过添加当前完全平方数i*i使得总和不超过n,并且这样的操作能减少之前记录的最少完全平方数的个数,就更新dp[j]的值为dp[j - i * i] + 1
  4. 返回结果:最后返回dp[n],即和为n时所需的最少完全平方数的个数。

注意

  • 代码注释中指出的“不需要这个if statement”,是因为在找完全平方数之和的问题中,任何正整数n都可以通过至少一个1^2(即至少一个1)来组合得到,所以直接尝试更新dp[j]的值而不需检查之前是否达到过是不可能的状态(即dp[j - i * i] != max的检查没有必要)。

综上,这段代码通过动态规划方法有效地求解了最少完全平方数之和的问题。

方法二:

class Solution {
    // 版本二, 先遍历背包, 再遍历物品
    public int numSquares(int n) {
        int max = Integer.MAX_VALUE;
        int[] dp = new int[n + 1];
        // 初始化
        for (int j = 0; j <= n; j++) {
            dp[j] = max;
        }
        // 当和为0时,组合的个数为0
        dp[0] = 0;
        // 遍历背包
        for (int j = 1; j <= n; j++) {
            // 遍历物品
            for (int i = 1; i * i <= j; i++) {
                dp[j] = Math.min(dp[j], dp[j - i * i] + 1);
            }
        }
        return dp[n];
    }
}

这段Java代码是解决“完全平方数”的另一个动态规划实现版本,目标依然是找出最少数量的完全平方数(如1, 4, 9, 16…)之和,使得这个和等于给定的正整数n。与第一个版本的主要区别在于遍历的顺序:这里是“先遍历背包,再遍历物品”。

代码解析

  1. 初始化:与第一个版本相同,首先创建一个长度为n+1的数组dp,其中dp[j]表示和为j时所需的最少完全平方数的个数。初始化所有dp[j]Integer.MAX_VALUE,然后设置dp[0]=0,表示和为0时不需要任何完全平方数。

  2. 遍历顺序改变

    • 外层循环现在遍历背包容量(从1到n),用j表示当前考虑的总和。
    • 内层循环遍历所有可能的完全平方数(从1^2到刚好不超过当前总和j的完全平方数),用i表示当前完全平方数的根。这确保了每次内循环都是对一个有效的完全平方数进行操作,不会超出背包容量。
  3. 状态转移:对于每个背包容量j,遍历所有小于等于它的完全平方数i*i,并尝试将当前完全平方数加入组合中(即从dp[j - i * i]转移而来),然后通过Math.min()函数更新dp[j]为已知的最小组合数。

  4. 返回结果:最后返回dp[n],即和为n时,所需最少的完全平方数个数。

优缺点

  • 优点:这种“先遍历背包,再遍历物品”的方式直接反映了背包问题的经典解法,逻辑上清晰地表达了对于每个总和j,尝试用所有可能的完全平方数去填充它,寻找最小组合数。
  • 缺点:在这个特定问题上,两种遍历顺序(先物品后背包 vs. 先背包后物品)在逻辑复杂度和效率上并无本质区别,主要取决于个人理解偏好。实际上,由于完全平方数的特殊性(连续的完全平方数之间差距增大),遍历顺序对性能的影响相对较小。

总之,这个版本提供了解决“完全平方数”问题的另一种动态规划实现思路,关键在于遍历顺序的调整,但核心的动态规划思想和状态转移方程保持一致。

139. 单词拆分

给你一个字符串 s 和一个字符串列表 wordDict 作为字典。如果可以利用字典中出现的一个或多个单词拼接出 s 则返回 true。

注意:不要求字典中出现的单词全部都使用,并且字典中的单词可以重复使用。

示例 1:

输入: s = “leetcode”, wordDict = [“leet”, “code”]
输出: true
解释: 返回 true 因为 “leetcode” 可以由 “leet” 和 “code” 拼接成。
示例 2:

输入: s = “applepenapple”, wordDict = [“apple”, “pen”]
输出: true
解释: 返回 true 因为 “applepenapple” 可以由 “apple” “pen” “apple” 拼接成。
注意,你可以重复使用字典中的单词。
示例 3:

输入: s = “catsandog”, wordDict = [“cats”, “dog”, “sand”, “and”, “cat”]
输出: false
在这里插入图片描述

方法一:

class Solution {
    public boolean wordBreak(String s, List<String> wordDict) {
        HashSet<String> set = new HashSet<>(wordDict);
        boolean[] valid = new boolean[s.length() + 1];
        valid[0] = true;

        for (int i = 1; i <= s.length(); i++) {
            for (int j = 0; j < i && !valid[i]; j++) {
                if (set.contains(s.substring(j, i)) && valid[j]) {
                    valid[i] = true;
                }
            }
        }

        return valid[s.length()];
    }
}

这段Java代码是一个解决方案,用于解决“单词拆分”问题。给定一个字符串s和一个字典wordDict(单词列表),判断字符串s是否可以被空格拆分成一个或多个字典中的单词。这是一个典型的动态规划问题。

代码解析

  1. 数据结构转换:首先,将wordDict转换为哈希集合HashSet<String> set,这样可以以O(1)的时间复杂度查询一个字符串是否在字典中。

  2. 初始化:定义一个布尔型数组valid,长度为s.length() + 1,其中valid[i]表示字符串s的前i个字符组成的子串是否可以被拆分成字典中的单词。初始化valid[0]true,因为空字符串是可以“拆分”的。

  3. 动态规划填充:外层循环从1遍历到s.length(),代表当前正在检查的子串的结束位置。内层循环从0遍历到当前的结束位置i,这是为了找到所有可能的前缀子串。如果存在某个前缀子串(从索引ji)在字典集合中,并且这个前缀子串的前一个位置(即j)的子串也是合法的(valid[j]true),那么将当前位置i标记为合法(valid[i] = true)。这里使用了!valid[i]作为提前终止的条件,一旦找到一个合法的拆分方式就不再继续查找,提高了效率。

  4. 返回结果:最后返回valid[s.length()],即整个字符串s是否可以被成功拆分。

示例

假设s = "leetcode"wordDict = ["leet", "code"],该函数将返回true,因为可以将s拆分成"leet"和"code",这两个单词都在字典中。

这段代码通过动态规划有效地解决了单词拆分问题,具有较好的时间和空间效率。

方法二:

// 另一种思路的背包算法
class Solution {
    public boolean wordBreak(String s, List<String> wordDict) {
        boolean[] dp = new boolean[s.length() + 1];
        dp[0] = true;

        for (int i = 1; i <= s.length(); i++) {
            for (String word : wordDict) {
                int len = word.length();
                if (i >= len && dp[i - len] && word.equals(s.substring(i - len, i))) {
                    dp[i] = true;
                    break;
                }
            }
        }

        return dp[s.length()];
    }
}

这段代码同样采用动态规划的思路来解决“单词拆分”问题,但实现方式稍有不同,具体解析如下:

算法思路

  • 初始化:创建一个布尔型数组dp,长度为s.length() + 1,其中dp[i]表示字符串s的前i个字符组成的子串是否能被字典中的单词组合覆盖。初始化dp[0] = true,表示空字符串是可以被任何词典中的单词组合覆盖的。

  • 双重循环遍历

    • 外层循环从1遍历到s.length(),用i表示当前考虑的子串的结束位置。
    • 内层循环遍历字典wordDict中的每个单词word
      • 计算当前单词的长度len
      • 判断当前子串的起始位置是否允许截取长度为len的子串,即i >= len,同时检查前len个字符组成的子串(即s.substring(i - len, i))是否与当前单词相等,并且这个子串的前一个位置的子串是否能被词典中的单词组合覆盖(即dp[i - len]true)。
      • 如果上述条件满足,说明找到了一个匹配的单词,可以将当前位置i标记为true并跳出内层循环,因为一旦找到一个合法的拆分方式就没有必要继续检查当前i的其他单词了(利用了“一旦满足条件即可结束”的剪枝优化)。
  • 返回结果:最后返回dp[s.length()],表示整个字符串s是否可以被字典中的单词组合覆盖。

优势与特点

  • 剪枝优化:通过在找到符合条件的单词后立即中断内层循环,减少了不必要的循环次数,提高了算法效率。
  • 直观易懂:代码直接体现了对每个子串尝试匹配字典中单词的过程,逻辑较为直观。
  • 空间效率:此方法仅使用了一个长度等于字符串长度加一的布尔数组,空间复杂度为O(n),其中n为字符串s的长度,与题目给定的字典大小无关,较为高效。

综上所述,这是一种有效且易于理解的动态规划解法,适用于解决给定字符串是否能被字典中的单词拆分的问题。

方法三:

// 回溯法+记忆化
class Solution {
    private Set<String> set;
    private int[] memo;
    public boolean wordBreak(String s, List<String> wordDict) {
        memo = new int[s.length()];
        set = new HashSet<>(wordDict);
        return backtracking(s, 0);
    }

    public boolean backtracking(String s, int startIndex) {
        // System.out.println(startIndex);
        if (startIndex == s.length()) {
            return true;
        }
        if (memo[startIndex] == -1) {
            return false;
        }

        for (int i = startIndex; i < s.length(); i++) {
            String sub = s.substring(startIndex, i + 1);
	    // 拆分出来的单词无法匹配
            if (!set.contains(sub)) {
                continue;                
            }
            boolean res = backtracking(s, i + 1);
            if (res) return true;
        }
        // 这里是关键,找遍了startIndex~s.length()也没能完全匹配,标记从startIndex开始不能找到
        memo[startIndex] = -1;
        return false;
    }
}

这段代码提供了一个使用回溯法加记忆化的解决方案来解决“单词拆分”问题。给定一个字符串s和一个单词字典wordDict,判断字符串s是否可以被空格拆分成一个或多个字典中的单词。以下是代码的详细解析:

类成员变量

  • set: 存储字典wordDict中的所有单词,使用HashSet以支持快速查找。
  • memo: 记忆化数组,用于存储字符串s的各个起始位置是否能够被拆分成字典中的单词。初始化为整型数组,长度与s相同,初始值默认为0,-1表示从该位置开始的子串不能被拆分。

方法解析

wordBreak 方法
  1. 初始化memo数组和set集合。
  2. 调用backtracking方法,从字符串的起始位置开始尝试拆分。
backtracking 方法
  • 参数:

    • s: 输入字符串。
    • startIndex: 当前开始拆分的位置索引。
  • 目的:

    • 递归地尝试从startIndex开始的子串是否能被拆分成字典中的单词。
  • 逻辑:

    • 基础情况:如果startIndex等于字符串长度,说明已经成功拆分到末尾,返回true
    • 记忆化检查:如果memo[startIndex]为-1,说明从startIndex开始的子串已经被探索过且不可拆分,直接返回false
    • 遍历:从startIndex到字符串结尾,逐步尝试截取子串,并检查该子串是否在字典中。
      • 如果子串在字典中,递归调用backtracking(s, i + 1),尝试剩余部分能否拆分。
      • 如果剩余部分可以被拆分,则当前子串也能被拆分,返回true
    • 回溯:如果所有尝试都无法成功拆分,标记memo[startIndex] = -1,表示从这个位置开始的子串不能被拆分,然后返回false

关键点

  • 记忆化搜索:通过memo数组避免重复计算,提高效率。
  • 回溯:在尝试失败后记录失败信息,防止同一子问题的重复探索。

这种方法在处理较长字符串和较大字典时,相较于简单的递归或暴力搜索能显著提升效率,但消耗的空间也会相应增加,主要是由于记忆化数组的使用。

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

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

相关文章

Qt:8.QWidget属性介绍(focuspolicy属性-控件焦点、stylesheet属性-为控件设置样式)

目录 一、focuspolicy属性-控件焦点&#xff1a; 1.1focuspolicy属性介绍&#xff1a; 1.2设置焦点策略——setFocusPolicy()&#xff1a; 1.3获取控件的焦点策略——focusPolicy()&#xff1a; 二、stylesheet属性——为控件设置样式&#xff1a; 2.1 stylesheet属性介绍…

虚拟机网络配置(静态网络)

解决问题&#xff1a;VMware中创建centOS虚拟机后使用ifconfig没有ip地址&#xff0c;但我想在主机&#xff08;Windows&#xff09;系统下使用shell连接虚拟机从而方便后续交互。 VMware中编辑->虚拟网络编辑器 &#xff08;注意需要管理员身份不然会无法修改&#xff09;…

Python容器 之 列表--定义

1.什么是列表呢&#xff1f; 列表(list)是 Python 中使用最频繁的数据类型, 在其他语言中通常叫做数组, 专门用来存储一组数据 列表,list, 使用 [ ] 列表可以存放任意多个数据 列表中可以存放任意类型的数据 列表中数据之间 使用 逗号隔开 2.列表如何定义&#xff1f; &#…

Android Compose 十二:常用组件列表 上拉加载

列表 上拉加载 当前思路 判断 列表最后一个显示的条目 为 数据集合的长度-1 用来记录刷新状态 var refreshing by remember {mutableStateOf(false)}数据集合 val list remember{List(10){"条目》》${it}"}.toMutableStateList()}用来记录列表当前状态及状态变化…

切片的基础知识

文章目录 ● Slice 的底层实现原理&#xff1f;● array 和 Slice 的区别&#xff1f;● 拷贝大切片一定比小切片代价大吗&#xff1f;● Slice 深拷贝和浅拷贝&#xff1f;● 零切片、空切片、nil切片&#xff1f;● Slice 的扩容机制&#xff1f;● Slice 为什么不是线程安全…

HarmonyOS Next开发学习手册——Native XComponent

场景介绍 Native XComponent是XComponent组件提供在Native层的实例&#xff0c;可作为JS层和Native层XComponent绑定的桥梁。XComponent所提供的NDK接口都依赖于该实例。接口能力包括获取Native Window实例、获取XComponent的布局/事件信息、注册XComponent的生命周期回调、注…

理性决策的艺术:从购房到择偶的数学智慧;37% 规则,做出最佳决策的秘诀;用数学模型解决人生难题

在面对人生重大决策时&#xff0c;如购房或择偶&#xff0c;我们常常感到迷茫和困惑。然而&#xff0c;如果我们能够将这些看似复杂的问题简化为数学模型&#xff0c;我们就能以更加理性和系统的方式做出决策。 37%规则 1950年代&#xff0c;当时几位数学家开始研究这样一个问…

钉钉开放AI生态战略的真正价值到底是什么?很多人都没看懂

来源&#xff1a; 首席数智官 hello 大家好&#xff0c;我们是数字化领军者都在看的首席数智官。 关注我&#xff0c;每天给你讲一个商业案例。 今天我们要给你讲的是&#xff1a;钉钉开放AI大模型生态的战略意义到底是什么&#xff1f; 「谁先赢得苹果&#xff0c;谁就赢得…

C++实现简化版Qt的QObject(3):增加父子关系、属性系统

前几天写了文章&#xff1a; C实现一个简单的Qt信号槽机制 C实现简化版Qt信号槽机制&#xff08;2&#xff09;&#xff1a;增加内存安全保障 之后感觉还不够过瘾&#xff0c;Qt中的QObject体系里还有不少功能特性没有实现。为了提高QObject的还原度&#xff0c;今天我们将父子…

欢迎回家!揭秘“嫦娥六号”背后的守望者

6月25日&#xff0c;嫦娥六号返回器携带来自月背的月球样品安全着陆在内蒙古四子王旗预定区域。这是时隔3年多后&#xff0c;中国探月工程的又一关键节点任务&#xff0c;也是时隔5年多后&#xff0c;嫦娥探测器再去月球背面。 在此次任务中&#xff0c;同元软控数字伴飞团队为…

C++使用Poco库封装一个HTTP客户端类

0x00 前言 我们在使用HTTP协议获取接口数据时&#xff0c;通常需要在Header和Query中添加参数&#xff0c;还有一种就是在Body中追加XML或者JSON格式的数据。本文主要讲述使用Poco库提交HTTP Post请求的Body中附加XML格式的数据&#xff0c;JSON格式的数据类似。 0x01 HttpCl…

Zuul介绍

Zuul 是 Netflix 开源的一个云平台网络层代理&#xff0c;它主要用于路由、负载均衡、中间件通信和动态路由。Zuul 本质上是一个基于 JVM 的网关&#xff0c;它提供了以下功能&#xff1a; 1.路由&#xff1a;Zuul 允许客户端和服务器之间的所有入站和出站请求通过一个中心化的…

Hadoop页面报错Permission denied: user=dr.who, access....

1、临时解决 hdfs dfs -chmod -R 777 /这种方法&#xff0c;存在一个不足&#xff0c;就是后面重新创建的文件夹&#xff0c;页面进行删除的时候&#xff0c;依然报这个错。 但是&#xff0c;对于应付紧急客户需求&#xff0c;可以临时用一下。 2、永久解决 查看页面的Owner…

超声波清洗机怎么选?极力推荐四款口碑大牌超声波清洗机

相信大家都知道超声波清洗机&#xff0c;每次眼镜脏的时候&#xff0c;去眼镜店里让老板帮忙清洗&#xff0c;她们用的就是超声波清洗机&#xff0c;通过超声波的原理深入物品深处清洁&#xff0c;清洁效果非常好。相对手洗的方式&#xff0c;超声波清洗机能够保护镜片在清洗过…

SpringCloud_Ribbon负载均衡

概述 SpringCloud底层其实是利用了一个名为Ribbon的组件&#xff0c;来实现负载均衡功能的。 源码 LoadBalancerInterceptor 其中含有intercept方法&#xff0c;拦截用户的HttpRequest请求&#xff1a; request.getURI() 获取请求uri&#xff0c;即http://userservice/use…

【哈哈大一上学的全忘了,重开!!】STM32从零入门物联网开发

本笔记资料来源 &#xff1a;STM32物联网入门30步&#xff1d;单片机物联网入门教程 WIFI连接阿里云物联网CubeMXHAL库蓝牙ESP8266杜洋主讲_哔哩哔哩_bilibili IOT&#xff1a;Internet of things 学习目标&#xff1a; 1.掌握洋桃IoT开发板的各功能以及驱动与基本应用 2.掌…

Python容器 之 字符串--字符串的常用操作方法

1.字符串查找方法 find() 说明&#xff1a;被查找字符是否存在于当前字符串中。 格式&#xff1a;字符串.find(被查找字符) 结果&#xff1a;如果存在则返回第一次出现 被查找字符位置的下标 如果不存在则返回 -1 需求&#xff1a; 1. 现有字符串数据: 我是中国人 2. 请设计程序…

vue根据文字长短展示跑马灯效果

介绍 为大家介绍一个我编写的vue组件 auto-marquee &#xff0c;他可以根据要展示文本是否超出展示区域&#xff0c;来判断是否使用跑马灯效果&#xff0c;效果图如下所示 假设要展示区域的宽度为500px&#xff0c;当要展示文本的长度小于500px时&#xff0c;只会展示文本&…

mysql5.7.30忘记root密码

windows系统安装了mysql5.7.30&#xff0c;在使用navicat链接mysql时候&#xff0c;提示 如何解决&#xff1a; 打开任务管理器的服务&#xff0c;查看有没有MYSQL服务。 如果没有&#xff0c;则按照下面的csdn博客进行操作。 https://blog.csdn.net/clj198606061111/article…

PostgreSQL的学习心得和知识总结(一百四十七)|深入理解PostgreSQL数据库之transaction chain的使用和实现

目录结构 注&#xff1a;提前言明 本文借鉴了以下博主、书籍或网站的内容&#xff0c;其列表如下&#xff1a; 1、参考书籍&#xff1a;《PostgreSQL数据库内核分析》 2、参考书籍&#xff1a;《数据库事务处理的艺术&#xff1a;事务管理与并发控制》 3、PostgreSQL数据库仓库…