暴力递归到动态规划(四)

news2024/12/21 22:55:06

请添加图片描述
⭐️前言⭐️

本篇文章是从暴力递归到动态规划篇目的最后一篇文章,包含了几道题目还有最终的大总结,相信这篇文章能让各位读者对动态规划有更深一步的了解。

🍉欢迎点赞 👍 收藏留言评论 📝私信必回哟😁

🍉博主将持续更新学习记录收获,友友们有任何问题可以在评论区留言

🍉博客中涉及源码及博主日常练习代码均已上传GitHub


请添加图片描述

📍内容导读📍

  • 🍅砍怪兽
  • 🍅最少货币数
  • 🍅拆分数字
  • 🍅数组拆分
  • 🍅数组拆分2
  • 🍅贴纸拼词
  • 🍅总结

🍅砍怪兽

题目:
给定3个参数,N,M,K
怪兽有N滴血,等着英雄来砍自己
英雄每一次打击,都会让怪兽流失[0~M]的血量
到底流失多少?每一次在[0~M]上等概率的获得一个值
求K次打击之后,英雄把怪兽砍死的概率

题解思路1:
总的情况数为(M+1)^k,通过暴力递归求出砍死怪兽的情况数,即可得到把怪兽砍死的概率。
递归终止的条件:
当次数为0时,看剩余血量,如果血量小于等于0,则返回1(说明是一种情况)
当血量为0时,剩余的次数都是能够砍死怪兽的情况,直接返回(M+1)^ times即为该情况下的所有情况数。
其余情况递归,根据0~M 血量的变化,进入不同的递归,记录不同的递归的情况数,求得最后的结果。

代码实现:

public class KillMonster {
    public static double right(int N,int M,int K) {
        if(N<1 || M<1 || K<1) {
            return 0;
        }
        long all=(long) Math.pow(M+1,K);
        long kill=process(K,M,N);
        return (double) ((double)kill/(double) all);
    }
    // 怪兽还剩hp点血
    // 每次的伤害范围为0~M
    // 还有times可以砍
    // 返回砍死的情况数
    public static long process(int times,int M,int hp) {
        if (times == 0) {
            return hp <= 0 ? 1 : 0;
        }
        if(hp<=0) {
            return (long) Math.pow(M+1,times);
        }
        long ways=0;
        for (int i = 0; i <=M ; i++) {
            ways+=process(times-1,M,hp-i);
        }
        return ways;
    }
}

题解思路2:
根据暴力递归来转化为动态规划解法,由于有两个可变参数,所以可以通过构建一个二维DP表来存储不同剩余次数、不同血量时的情况数,最后返回表中需要位置的结果即可。
(注意:当血量为负的时候,把怪兽砍死的结果数就是(M+1)^(times-1))

代码实现:

public class KillMonster {
    public static double dp1(int N,int M,int K) {
        if(N<1 || M<1 || K<1) {
            return 0;
        }
        long all=(long) Math.pow(M+1,K);
        long[][] dp=new long[K+1][N+1];
        dp[0][0]=1;
        for (int times = 1; times <=K ; times++) {
            dp[times][0]=(long) Math.pow(M+1,times);
            for (int hp = 1; hp <=N ; hp++) {
                long ways=0;
                for (int i = 0; i <=M ; i++) {
                    if(hp-i>=0) {
                        ways+=dp[times-1][hp-i];
                    }else {  // 当血量为负的时候,情况数直接就是下边的公式
                        ways+=(long) Math.pow(M+1,times-1);
                    }
                }
                dp[times][hp]=ways;
            }
        }
        long kill=dp[K][N];
        return (double) ((double)kill/(double) all);
    }
}

题解思路3:
由于在求每个位置的dp结果时,存在for循环的枚举依赖,所以可以通过严格依赖的表结构关系进行优化,不使用枚举依赖。

代码实现:

public class KillMonster {
    // 枚举优化
    public static double dp2(int N,int M,int K) {
        if(N<1 || M<1 ||K<1) {
            return 0;
        }
        long all=(long) Math.pow(M+1,K);
        long[][] dp=new long[K+1][N+1];
        dp[0][0]=1;
        for (int times = 1; times <=K ; times++) {
            dp[times][0]=(long) Math.pow(M+1,times);
            for (int hp = 1; hp <=N ; hp++) {
                dp[times][hp]=dp[times][hp-1]+dp[times-1][hp];
                if(hp-1-M>=0) {// 没有越界
                    dp[times][hp]-=dp[times-1][hp-1-M];
                }else {// 如果越界了说明血量为负,也得减去情况数
                    dp[times][hp]-=Math.pow(M+1,times-1);
                }
            }
        }
        long kill=dp[K][N];
        return (double) ((double)kill/(double) all);
    }
}

🍅最少货币数

题目:
arr是面值数组,其中的值都是正数且没有重复。再给定一个正数aim。
每个值都认为是一种面值,且认为张数是无限的。
返回组成aim的最少货币数

题解思路:
从左往右的尝试模型,依次考虑每个面值对应不同数的情况,记录能够组成aim的最小张数返回即可。

代码实现:

public class MinCoinsNoLimit {
    public static int minCoins(int[] arr,int aim) {
        return process(arr,0,aim);
    }
    
    // arr[index...]面值,每种面值张数自由选择
    // 凑出正好rest的钱数,返回最小张数
    // 拿Integer.MAX_VALUE标记怎样也凑不出
    public static int process(int[] arr,int index,int rest) {
        if(index==arr.length) {
            return rest==0?0:Integer.MAX_VALUE;
        }else {
            int ans=Integer.MAX_VALUE;
            for (int pages = 0; pages*arr[index] <=rest ; pages++) {
                int next=process(arr,index+1,rest-pages*arr[index]);
                if(next!=Integer.MAX_VALUE) {
                    ans=Math.min(ans,next+pages);
                }
            }
            return ans;
        }
    }
}

题解思路2:
根据暴力递归转化为动态规划,由于有两个可变参数,所以可以依此构建一个二维dp表,记录不同面值,不同rest,再根据index+1位置面值的不同张数返回的情况数,取最小结果填入。

代码实现:

public class MinCoinsNoLimit {
    public static int dp1(int[] arr,int aim) {
        if(aim==0) {
            return 0;
        }
        int N=arr.length;
        int[][] dp=new int[N+1][aim+1];
        dp[N][0]=0;
        for (int j = 1; j <=aim ; j++) {
            dp[N][j]=Integer.MAX_VALUE;
        }
        for (int index = N-1; index >=0 ; index--) {
            for (int rest = 0; rest <=aim ; rest++) {
                int ans=Integer.MAX_VALUE;
                for (int pages = 0; pages*arr[index] <=rest ; pages++) {
                    int next=dp[index+1][rest-pages*arr[index]];
                    if(next!=Integer.MAX_VALUE) {
                        ans=Math.min(ans,pages+next);
                    }
                }
                dp[index][rest]=ans;
            }
        }
        return dp[0][aim];
    }
}

题解思路3:
由于每个位置的填写,都依赖于枚举得到的结果,所以可以根据严格依赖的表结构位置关系,优化枚举行为。
在这里插入图片描述

红色方格的值依赖于Math.min(a+0,b+1,c+2,d+3) 【最后加几取决于rest,只要不越过左边界即可】
黄色方格的值依赖于Math.min(b+0,c+1,d+2)

所以dp[index][rest]=Math.min(dp[index+1][rest],dp[index][rest-arr[index]]+1)

代码实现:

public class MinCoinsNoLimit {
    public static int dp2(int[] arr,int aim) {
        if(aim==0) {
            return 0;
        }
        int N=arr.length;
        int[][] dp=new int[N+1][aim+1];
        dp[N][0]=0;
        for (int j = 1; j <=aim ; j++) {
            dp[N][j]=Integer.MAX_VALUE;
        }
        for (int index = N-1; index >=0 ; index--) {
            for (int rest = 0; rest <=aim ; rest++) {
                dp[index][rest]=dp[index+1][rest];
                if(rest-arr[index]>=0&&
                dp[index][rest-arr[index]]!=Integer.MAX_VALUE) {
                    dp[index][rest]=Math.min(dp[index][rest-arr[index]]+1,dp[index][rest]);
                }
            }
        }
        return dp[0][aim];
    }
}

🍅拆分数字

题目:
给定一个正数n,求n的裂开方法数,
规定:后面的数不能比前面的数小
比如4的裂开方法有:
1+1+1+1、1+1+2、1+3、2+2、4
5种,所以返回5

题解思路1:
如果想要拆分5,就假设拆分6,先拆成了1和5,看5剩下多少种拆分办法。
base case:当rest剩余0时,说明完全拆分完成,则返回一种情况,如果pre>rest,则不符合条件,返回0。

代码实现:

public class SplitNumber {
    // n为正数
    public static int ways(int n) {
        if(n<0) {
            return 0;
        }
        if(n==1) {
            return 1;
        }
        return process(1,n);
    }

    // 上一个拆出来的数是pre
    // 还剩rest需要去拆
    // 返回拆解的方法数
    public static int process(int pre, int rest) {
        if(rest==0) {
            return 1;
        }
        if(pre>rest) {
            return 0;
        }
        int ways=0;
        for (int first = pre; first <=rest ; first++) {
            ways+=process(first,rest-first);
        }
        return ways;
    }
}

题解思路2:
由暴力递归转化为动态规划,有两个可变参数pre和rest,先跟base case完成部分位置的填写,再根据暴力递归的依赖关系,来推断出dp表中其他位置该如何填写。
在这里插入图片描述

代码实现:

public class SplitNumber {
    public static int dp1(int n) {
        if(n<0) {
            return 0;
        }
        if(n==1) {
            return 1;
        }
        int[][] dp=new int[n+1][n+1];
        for (int pre = 1; pre <=n ; pre++) {
            dp[pre][0]=1;
            dp[pre][pre]=1;
        }
        for (int pre = n-1; pre >=0 ; pre--) {
            for (int rest = pre+1; rest <=n ; rest++) {
                int ways=0;
                for (int first = pre; first <=rest ; first++) {
                    ways+=dp[first][rest-first];
                }
                dp[pre][rest]=ways;
            }
        }
        return dp[1][n];
    }
}

题解思路3:
由于每个位置的填写有枚举依赖,所以可以根据严格依赖的位置关系再做优化,对枚举行为进行优化。
dp[pre][rest]=dp[pre+1][rest]+dp[pre][rest-pre]
在这里插入图片描述

代码实现:

public class SplitNumber {
    public static int dp2(int n) {
        if(n<0) {
            return 0;
        }
        if(n==1) {
            return 1;
        }
        int[][] dp=new int[n+1][n+1];
        for (int pre = 1; pre <=n ; pre++) {
            dp[pre][0]=1;
            dp[pre][pre]=1;
        }
        for (int pre = n-1; pre >=0 ; pre--) {
            for (int rest = pre+1; rest <=n ; rest++) {
                dp[pre][rest]=dp[pre+1][rest];
                dp[pre][rest]+=dp[pre][rest-pre];
            }
        }
        return dp[1][n];
    }
}

🍅数组拆分

题目:
给定一个正数数组arr,
请把arr中所有的数分成两个集合,尽量让两个集合的累加和接近
返回最接近的情况下,较小集合的累加和

题解思路1:
从左往右的尝试模型,每个位置都有要或者不要两种情况,递归就是返回接近于rest但不大于rest的最大累加和。
代码实现:

public class SplitSumClosed {
    public static int right(int[] arr) {
        if(arr==null||arr.length<2) {
            return 0;
        }
        int sum=0;
        for(int num:arr) {
            sum+=num;
        }
        return process(arr,0,sum/2);
    }

    // arr[i...]可以自由选择,返回累加和尽量接近rest,但不能超过rest的情况下,最接近的累加和是多少
    public static int process(int[] arr,int i,int rest) {
        if(i==arr.length) {
            return 0;
        }else { // 还有数
            // 可能性1,不使用arr[i]
            int p1=process(arr,i+1,rest);
            // 可能性2,使用arr[i]
            int p2=0;
            if(arr[i]<=rest) {
                p2=arr[i]+process(arr,i+1,rest-arr[i]);
            }
            return Math.max(p1,p2);// 返回两种可能性中更接近的
        }
    }
}

题解思路2:
暴力递归中有两个可变参数,所以可以优化为一个二维dp表,来进行存储不同状态下的结果,根据位置依赖关系来填写dp表,最后返回所需要状态的结果。
代码实现:

public class SplitSumClosed {
    public static int dp(int[] arr) {
        if(arr==null||arr.length<2) {
            return 0;
        }
        int sum=0;
        for (int num:arr) {
            sum+=num;
        }
        sum/=2;
        int N=arr.length;
        int[][] dp=new int[N+1][sum+1];
        for (int i = N-1; i >=0 ; i--) {
            for (int rest = 0; rest <=N ; rest++) {
                // 可能性1,不使用arr[i]
                int p1=dp[i+1][rest];
                // 可能性2,使用arr[i]
                int p2=0;
                if(arr[i]<=rest) {
                    p2=arr[i]+dp[i+1][rest-arr[i]];
                }
                dp[i][rest]= Math.max(p1,p2);
            }
        }
        return dp[0][sum];
    }
}

🍅数组拆分2

题目:
给定一个正数数组arr,请把arr中所有的数分成两个集合
如果arr长度为偶数,两个集合包含数的个数要一样多
如果arr长度为奇数,两个集合包含数的个数必须只差一个
请尽量让两个集合的累加和接近
返回最接近的情况下,较小集合的累加和

题解思路1:
与数组拆分1类似,只不过加上了个数限制,要根据arr长度来决定集合中只能选几个数。
代码实现:

public class SplitSumClosedSizeHalf {
    public static int right(int[] arr) {
        if(arr==null||arr.length<2) {
            return 0;
        }
        int sum=0;
        for (int num:arr) {
            sum+=num;
        }
        if((arr.length&1)==0) {
            return process(arr,0,arr.length/2,sum/2);
        }else {
            return Math.max(process(arr,0,arr.length/2,sum/2),process(arr,0,arr.length/2+1,sum/2));
        }
    }
    // arr[i...]自由选择,挑选的个数一定是picks个,累加和<=rest,离rest最接近的返回
    public static int process(int[] arr, int i, int picks, int rest) {
        if(i==arr.length) {
            return picks==0?0:-1;  // 不合法用-1标记
        }else {
            // 不用arr[i]
            int p1=process(arr, i+1, picks, rest);
            // 用arr[i]
            int p2=-1;
            int next=-1;
            if(arr[i]<=rest) {
                next=process(arr,i+1,picks-1,rest-arr[i]);
            }
            // 如果下一层有效
            if(next!=-1) {
                p2=arr[i]+next;
            }
            return Math.max(p1,p2);
        }
    }
}

题解思路2:
根据暴力递归的三个可变参数,建立一个三维的dp表,根据位置依赖关系填写dp表,最后返回所要位置的dp结果。
代码实现:

public class SplitSumClosedSizeHalf {
    public static int dp(int[] arr) {
        if(arr==null||arr.length<2) {
            return 0;
        }
        int sum=0;
        for (int num:arr) {
            sum+=num;
        }
        sum/=2;
        int N=arr.length;
        int M=(N+1)/2;// 向上取整
        int[][][] dp=new int[N+1][M+1][sum+1];
        for (int i = 0; i <= N; i++) {
            for (int j = 0; j <= M; j++) {
                for (int k = 0; k <= sum; k++) {
                    dp[i][j][k] = -1;
                }
            }
        }
        for (int rest = 0; rest <=sum ; rest++) {
            dp[N][0][rest]=0;
        }
        for (int i = N-1; i >=0 ; i--) {
            for (int picks = 0; picks <=M ; picks++) {
                for (int rest = 0; rest <=sum ; rest++) {
                    // 不用arr[i]
                    int p1=dp[i+1][picks][rest];
                    // 用arr[i]
                    int p2=-1;
                    int next=-1;
                    if(arr[i]<=rest) {
                        next=dp[i+1][picks-1][rest-arr[i]];
                    }
                    // 如果下一层有效
                    if(next!=-1) {
                        p2=arr[i]+next;
                    }
                    dp[i][picks][rest]= Math.max(p1,p2);
                }
            }
        }
        if((arr.length&1)==0) {
            return dp[0][arr.length/2][sum];
        }else {
            return Math.max(dp[0][arr.length/2][sum],dp[0][arr.length/2+1][sum]);
        }
    }
}

🍅贴纸拼词

题目:
给定一个字符串str,给定一个字符串类型的数组arr。
arr里的每一个字符串,代表一张贴纸,你可以把单个字符剪开使用,目的是拼出str来。
返回需要至少多少张贴纸可以完成这个任务。
例子:str= “babac”,arr = {“ba”,“c”,“abcd”}
至少需要两张贴纸"ba"和"abcd",因为使用这两张贴纸,把每一个字符单独剪开,含有2个a、2个b、1个c。是可以拼出str的。所以返回2。
https://leetcode-cn.com/problems/stickers-to-spell-word/

题解思路:
每张贴纸都可以作为第一张,如果该贴纸有效(即可以减掉部分字符),则递归进入下一层(传入的参数变为减后的字符串);如果该贴纸无效,则直接就是默认为系统最大值,最后返回记录的最小结果。

代码实现:

public class StickersToSpellWord {
    public static int minSticker1(String[] stickers,String target) {
        int ans=process(stickers,target);
        return ans==Integer.MAX_VALUE?-1:ans;
    }

    // 所有贴纸stickers,每一种贴纸都有无穷张
    // target
    // 最少张数
    public static int process(String[] stickers,String target) {
        if(target.length()==0) {
            return 0;
        }
        int min=Integer.MAX_VALUE;
        for (String first:stickers) {
            String rest=minus(target,first);
            if(rest.length()!=target.length()) {
                min=Math.min(min,process(stickers,rest));
            }
        }
        return min+(min==Integer.MAX_VALUE?0:1);
    }

    public static String minus(String s1,String s2) {
        char[] str1=s1.toCharArray();
        char[] str2=s2.toCharArray();
        int[] count=new int[26];
        for (char ch:str1) {
            count[ch-'a']++;
        }
        for (char ch:str2) {
            count[ch-'a']--;
        }
        StringBuilder builder=new StringBuilder();
        for (int i = 0; i < 26; i++) {
            if(count[i]>0) {
                for (int j = 0; j < count[i]; j++) {
                    builder.append((char) (i+'a'));
                }
            }
        }
        return builder.toString();
    }
}

题解思路2:
可以将词频表和贴纸都转换为数组,在相减时可以直接数组相减。

代码实现:

public class StickersToSpellWord {
    public static int minStickers2(String[] stickers,String target) {
        int N=stickers.length;
        // 关键优化(用词频表代替贴纸数组)
        int[][] counts=new int[N][26];
        for (int i = 0; i < N; i++) {
            char[] str=stickers[i].toCharArray();
            for (char ch:str) {
                counts[i][ch-'a']++;
            }
        }
        int ans=process2(counts,target);
        return ans==Integer.MAX_VALUE?-1:ans;
    }
    
    // stickers[i]数组
    // 每种贴纸都有无穷张
    // 返回搞定target的最少张数
    public static int process2(int[][] stickers,String t) {
        if(t.length()==0) {
            return 0;
        }
        // target做出词频统计
        char[] target=t.toCharArray();
        int[] tcounts=new int[26];
        for (char ch:target) {
            tcounts[ch-'a']++;
        }
        int N=stickers.length;
        int min=Integer.MAX_VALUE;
        for (int i = 0; i < N; i++) {
            // 尝试第一张贴纸是谁
            int[] sticker=stickers[i];
            // 剪枝优化
            if(sticker[target[0]-'a']>0) {  // 如果目标target中的字符在该贴纸中存在
                StringBuilder builder=new StringBuilder();
                for (int j = 0; j < 26; j++) {
                    if(tcounts[j]>0) {
                        int nums=tcounts[j]-sticker[j];
                        for (int k = 0; k < nums; k++) {
                            builder.append((char) j+'a');
                        }
                    }
                }
                String rest=builder.toString();
                min=Math.min(min,process2(stickers,rest));
            }
        }
        return min+(min==Integer.MAX_VALUE?0:1);
    }
}

🍅总结

在这里插入图片描述
什么暴力递归可以继续优化?
有重复调用同一个子问题的解,这种递归可以优化;
如果每个子问题都是不同的解,无法优化也不用优化。

暴力递归和动态规划的关系
某一个暴力递归,有解的重复调用,就可以把这个暴力递归优化成动态规划
任何动态规划问题,都一定对应着某一个有解的重复调用的暴力递归
但不是所有的暴力递归,都一定对应着动态规划

常见的四种模型
1、从左往右的尝试模型
2、范围尝试模型
3、样本对应模型
4、业务限制模型


⭐️最后的话⭐️
总结不易,希望uu们不要吝啬你们的👍哟(^U^)ノ~YO!!如有问题,欢迎评论区批评指正😁

请添加图片描述

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

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

相关文章

使用大型语言模(LLM)构建系统(五):输出结果检查

今天我学习了DeepLearning.AI的 Building Systems with LLM 的在线课程&#xff0c;我想和大家一起分享一下该门课程的一些主要内容。今天我们来学习输出结果检查。输出结果检查包含以下两部分内容&#xff1a; 检查输出是否存在潜在有害内容检查输出是否基于提供的产品信息 下…

easyrecovery2023永久免费版激活密钥,手把手教您用EasyRecovery快速恢复数据

Ontrack EasyRecovery Crack Professional是一个全面的备份和恢复实用程序&#xff0c;可以从多个数据丢失事件中恢复文件&#xff0c;例如常见的意外删除、更严重的&#xff08;有时是病毒引起的&#xff09;分区或驱动器格式化&#xff0c;甚至硬盘严重损坏后的数据丢失。免费…

Python实现ACO蚁群优化算法优化LightGBM回归模型(LGBMRegressor算法)项目实战

说明&#xff1a;这是一个机器学习实战项目&#xff08;附带数据代码文档视频讲解&#xff09;&#xff0c;如需数据代码文档视频讲解可以直接到文章最后获取。 1.项目背景 蚁群优化算法(Ant Colony Optimization, ACO)是一种源于大自然生物世界的新的仿生进化算法&#xff0c…

全球计算科学和人工智能领域Altair

作为全球计算科学和人工智能领域的领导者 Altair,全新数据分析与人工智能平台 Altair RapidMiner。此次启动仪式以 “数据科学&#xff0c;解码智能未来” 为主题。目的是为了更好的助力本土用户推进高效数字化转型&#xff0c;实现数据分析与人工智能应用的降本增效. 作为全球…

PR视频导出时出现编译影片出错的提示怎么解决?

大家使用PR软件的时候因为版本不兼容等各种原因&#xff0c;导致剪辑好的视频导不出来&#xff08;如下图所示&#xff09;&#xff0c;这种情况有两种方法可以解决。&#xff08;我用的版本是2022Pro&#xff0c;电脑需要Win10的系统&#xff0c;如有需要下载的&#xff0c;可…

使用 Terraform 在 GCP 上一键部署 EMQX MQTT Broker

引言 MQTT 是一种轻量级的消息传递协议&#xff0c;适用于物联网应用实现设备间通信。 作为一款主流的开源 MQTT Broker&#xff0c;EMQX 能够提供高扩展性、可靠性和安全性的 MQTT 消息传递服务。 借助广泛应用的基础设施即代码&#xff08;IaC&#xff09;工具 Terraform&a…

外包干了4年,人直接废了···

有一说一&#xff0c;外包没有给很高的薪资&#xff0c;是真不能干呀&#xff01; 先说一下自己的情况&#xff0c;大专生&#xff0c;19年通过校招进入湖南某软件公司&#xff0c;干了接近4年的功能测试&#xff0c;今年年初&#xff0c;感觉自己不能够在这样下去了&#xff0…

【javaScript】Proxy与Object.defineProperty的区别

目录 功能方面的区别&#xff1a;Object.defineProperty对整个对象进行赋值&#xff0c;不会触发set拦截Object.defineProperty对整个对象进行遍历&#xff0c;不会触发get拦截Proxy自定义拦截行为 Proxy的优缺点优点&#xff1a;缺点 Proxy和Object.defineProperty都是JavaScr…

亿发软件:中医药信息一体化解决方案,智慧中药房煎配系统

智慧中药房是中医药服务的一种新模式&#xff0c;随着中医药信息化建设逐渐兴起&#xff0c;湖南远跃顺应时代发展需求&#xff0c;深度分析中医药行业现状&#xff0c;充分发挥研发技术创新&#xff0c;融合运用互联网、物联网、大数据等技术创新成果&#xff0c;研发出中药信…

上市后首份年报,读书郎去年净利下滑94%

又是一年高考逐梦时&#xff0c;寒窗二十四年的读书郎如何克服“偏科”症&#xff1f; 近日&#xff0c;读书郎教育控股有限公司&#xff08;下称“读书郎”,02385.HK)交出了上市后的首份年报。 3月29日&#xff0c;财报公布的首个交易日&#xff0c;其股价下跌0.27%&#xf…

起点,而非终点——我的创作纪念日

机缘 我注定遇见CSDN. 作为一位oier,在学习算法,查找题解的时候&#xff0c;第一个看到的&#xff0c;总是CSDN。 于是&#xff0c;我加入了CSDN这个家庭。 茫茫然然&#xff0c;带着无限的憧憬&#xff0c;我写下了第一篇文章&#xff1a;《你好&#xff0c;CSDN》 我的C…

什么是融合CDN?融合CDN的优势和常见的调度模式有哪些?

什么是融合CDN&#xff1f; 为了理解什么是融合CDN&#xff0c;我们先了解什么是CDN。CDN是一个地理分布的边缘服务器网络&#xff0c;其目标是提供更快、可靠的互联网内容交付。CDN通过在网络边缘缓存互联网内容来实现这一点&#xff0c;从而减少数据源和最终用户之间的物理距…

“入职半年,绩效背了C”,谈谈「绩效管理」的误区及对策

职场打工人上班吃维C&#xff0c;绩效却背C&#xff0c;苦C久矣&#xff01; 又到年中“考核季”&#xff0c;被绩效考核折磨到“头秃”的职场打工人们&#xff0c;选择将情绪释放到社交平台。 一位朋友分享了自己绩效被打C的经历&#xff1a;“茶饭不思&#xff0c;感觉自己很…

数学建模经历-程序人生

引言 即将大四毕业(现在大三末)&#xff0c;闲来无事(为了冲粽子)就写一篇记录数学建模经历的博客吧。其实经常看到一些大佬的博客里会有什么"程序人生"、"人生感想"之类的专栏&#xff0c;但是由于我只是一个小趴菜没什么阅历因此也就没有写过类似的博客…

一文详解IPv4与IPv6协议

前段时间的工作大多与通信协议相关&#xff0c;随着协议相关工作的不断深入&#xff0c;相关数据包的分析占据了不少工作时间。在数据报文分析中&#xff0c;发现大学期间IP协议内容已经重新还给了老师&#xff0c;相关知识完全没有了印象&#xff0c;这篇文章算是一篇复习文&a…

Day05 04-MySQL分库分表介绍

文章目录 第十七章 MySQL分库分表17.1 什么是分库分表17.2 为什么要分库分表17.3 垂直切分17.3.1 垂直分库17.3.2 垂直分表 17.4 水平切分17.4.1 水平分库17.4.2 水平分表17.4.3 常见的水平切分规则 第十七章 MySQL分库分表 17.1 什么是分库分表 MySQL数据库常见的优化方案中…

新的管理中心应用

大家好&#xff0c;才是真的好。 Notes/Domino 14 Drop1如约而至&#xff0c;让我们有了很多内容要写。首先是管理新特性&#xff0c;就像上篇和大家介绍的&#xff0c;感受最直接的就是新的管理中心应用AdminCentral。 以目前的Drop1环境来看&#xff0c;管理中心应用主要功…

plus.runtime.version总是13.8.4

引言 最近在uniapp中使用到了plus.runtime.version&#xff0c;但是在开发环境下一直无法获取到真正的版本号&#xff0c;他的值一直都是13.8.4&#xff0c;在全局进行搜索也没有发现哪里设置了13.8.4&#xff0c;后来查阅了相关资料才知道这并不是自己写错了。 场景复现&…

你知不知道ai如何绘画的

有没有想要自己能绘画一幅有艺术感的画呢&#xff1f;想自己也能将自己所想的事、物或景绘画出来呢&#xff1f;其实你不用担心你的绘画功底不行&#xff0c;因为现在有一门ai绘画的技术&#xff0c;它可以将你提供的信息用智能技术自动绘画&#xff0c;你也可以把它当做一门新…

数字IC前端学习笔记:仲裁轮询(一)

相关文章 数字IC前端学习笔记&#xff1a;LSFR&#xff08;线性反馈移位寄存器&#xff09; 数字IC前端学习笔记&#xff1a;跨时钟域信号同步 数字IC前端学习笔记&#xff1a;信号同步和边沿检测 数字IC前端学习笔记&#xff1a;锁存器Latch的综合 数字IC前端学习笔记&am…