【力扣周赛】第 361 场周赛(⭐前缀和+哈希表 树上倍增、LCA⭐)

news2024/11/17 3:31:47

文章目录

  • 竞赛链接
  • Q1:7020. 统计对称整数的数目
    • 竞赛时代码——枚举预处理
  • Q2:8040. 生成特殊数字的最少操作(倒序遍历、贪心)
    • 竞赛时代码——检查0、00、25、50、75
  • Q3:2845. 统计趣味子数组的数目
    • 竞赛时代码——前缀和+哈希表
    • 相似题目——1590. 使数组和能被 P 整除(确实很相似的题目)
  • Q4:2846. 边权重均等查询⭐⭐⭐⭐⭐
    • 读题
    • 解法——树上倍增、最近公共祖先LCA
    • 相关题目
  • 成绩记录

竞赛链接

https://leetcode.cn/contest/weekly-contest-361/

Q1:7020. 统计对称整数的数目

https://leetcode.cn/problems/count-symmetric-integers/

在这里插入图片描述
提示:
1 <= low <= high <= 10^4

竞赛时代码——枚举预处理

预处理所有数字是否为对称整数。
cnt[i]表示 <=i 的数字中有几个对称整数。

class Solution {
    static int[] cnt = new int[10005];
    // 预处理
    static {
        for (int i = 1; i <= 10001; ++i) {
            cnt[i] = cnt[i - 1];
            if (op(i)) cnt[i]++;
        }
    }
    
    public int countSymmetricIntegers(int low, int high) {
        return cnt[high] - cnt[low - 1];
    }
    
    // 判断x是否为对称整数
    public static boolean op(int x) {
        List<Integer> ls = new ArrayList<>();
        while (x != 0) {
            ls.add(x % 10);
            x /= 10;
        }
        int n = ls.size();
        if (n % 2 == 1) return false;
        int a = 0, b = 0;
        for (int i = 0; i < n; ++i) {
            if (i < n / 2) a += ls.get(i);
            else b += ls.get(i);
        }
        return a == b;
    }
}

Q2:8040. 生成特殊数字的最少操作(倒序遍历、贪心)

https://leetcode.cn/problems/minimum-operations-to-make-a-special-number/

在这里插入图片描述
提示

1 <= num.length <= 100
num 仅由数字 '0' 到 '9' 组成
num 不含任何前导零

竞赛时代码——检查0、00、25、50、75

检查位置最靠后的 00、25、50、75 的位置。

如果都不存在但是有 0 的话,答案则为 n - 1。(因为 0 可以不删)

class Solution {
    public int minimumOperations(String num) {
        int n = num.length();
        boolean f0 = false, f5 = false;
        for (int i = n - 1; i >= 0; --i) {
            char ch = num.charAt(i);
            if (ch == '0') {
                if (f0) return n - i - 2;       // 检查00
                f0 = true;
            } else if (ch == '5') {
                if (f0)  return n - i - 2;;     // 检查50
                f5 = true;
            } else if (ch == '2' || ch == '7') {
                if (f5)  return n - i - 2;;     // 检查25,75
            }
        }
        if (f0) return n - 1;                   // 检查是否有0
        return n;
    }
}

Q3:2845. 统计趣味子数组的数目

https://leetcode.cn/problems/count-of-interesting-subarrays/

在这里插入图片描述

提示:
1 <= nums.length <= 10^5
1 <= nums[i] <= 10^9
1 <= modulo <= 10^9
0 <= k < modulo

竞赛时代码——前缀和+哈希表

使用前缀和数组可以快速求出从 l ~ r 之间满足要求的元素个数 cnt。

求出前缀和数组之后,从前往后依次枚举下标。对于当前的前缀和 sum[r],前面有若干个满足 (sum[r] - sum[x]) % modulo == k 的下标,这些下标的共同特征是:它们的值 sum[x] = (sum[r] - k + modulo) % modulo。
在枚举的过程中用哈希表 cnt 记录下满组 sum[i] = y 的 i 的数量,就可以快速找到 r 之前有几个可以和当前下标配对的下标 l。

class Solution {
    public long countInterestingSubarrays(List<Integer> nums, int modulo, int k) {
        int n = nums.size();
        int[] x = new int[n], sum = new int[n + 1];		// x是原始数组,sum是前缀和数组
        Map<Integer, Integer> cnt = new HashMap<>();    // 存储各个余数为key的位置的数量
        cnt.put(0, 1);
        long ans = 0;
        
        for (int i = 0; i < n; ++i) {
            if (nums.get(i) % modulo == k) x[i] = 1;
            sum[i + 1] = (sum[i] + x[i]) % modulo;      // 前缀和

            int r = sum[i + 1];
            ans += cnt.getOrDefault((r - k + modulo) % modulo, 0);
            cnt.merge(r, 1, Integer::sum);
        }
        return ans;
    }
}

相似题目——1590. 使数组和能被 P 整除(确实很相似的题目)

https://leetcode.cn/problems/make-sum-divisible-by-p/description/

在这里插入图片描述

提示:

1 <= nums.length <= 105
1 <= nums[i] <= 109
1 <= p <= 109

class Solution {
    public int minSubarray(int[] nums, int p) {
        int n = nums.length;
        int[] sum = new int[n + 1];                     // 前缀和数组
        for (int i = 0; i < n; ++i) {
            sum[i + 1] = (sum[i] + nums[i]) % p;
        }
        int t = sum[n], ans = n;
        if (t == 0) return 0;
        Map<Integer, Integer> idx = new HashMap<>();    // 记录各个前缀和出现的下标
        idx.put(0, -1);
        for (int i = 0; i < n; ++i) {
            // x是当前前缀和,y是和x配对组成t的前缀和
            int x = sum[i + 1], y = (x - t + p) % p;   
            // 如果之前有y,就尝试更新答案 
            if (idx.containsKey(y)) ans = Math.min(ans, i - idx.get(y));
            idx.put(x, i);
        }
        return ans == n? -1: ans;
    }
}

Q4:2846. 边权重均等查询⭐⭐⭐⭐⭐

https://leetcode.cn/problems/minimum-edge-weight-equilibrium-queries-in-a-tree/

在这里插入图片描述
在这里插入图片描述

提示:
1 <= n <= 10^4
edges.length == n - 1
edges[i].length == 3
0 <= ui, vi < n
1 <= wi <= 26
生成的输入满足 edges 表示一棵有效的树
1 <= queries.length == m <= 2 * 10^4
queries[i].length == 2
0 <= ai, bi < n

读题

给了一个 n 个节点的无向图。

每次查询,给两个点 a ,b。求 a 和 b 路径之间的所有边权,都变成相等需要操作几步——实际上是求 a 和 b 之间有几条边,其中出现次数最多的边权出现了几次。

解法——树上倍增、最近公共祖先LCA

https://leetcode.cn/problems/minimum-edge-weight-equilibrium-queries-in-a-tree/solutions/2424060/lca-mo-ban-by-endlesscheng-j54b/
在这里插入图片描述
思路总结:
用树上倍增的思想维护:各个节点的深度、各个节点和父节点之间各种边权的数量。

求答案时,先将两个节点放在同一深度,实现方法是 y 先跳 d[y] - d[x] 的深度。
然后,x 和 y 一起往上跳。

class Solution {
    public int[] minOperationsQueries(int n, int[][] edges, int[][] queries) {
        // 临界表存储无向图
        List<int[]>[] g = new ArrayList[n];             
        Arrays.setAll(g, e -> new ArrayList<>());
        for (int[] e: edges) {
            int x = e[0], y = e[1], w = e[2] - 1;
            g[x].add(new int[]{y, w});
            g[y].add(new int[]{x, w});
        }

        int m = 32 - Integer.numberOfLeadingZeros(n);   // n的二进制长度
        int[][] pa = new int[n][m];     // pa[x][i]表示节点x的第2^i个父节点
        for (int i = 0; i < n; ++i) {
            Arrays.fill(pa[i], -1);     // -1表示没有这个父节点
        }
        int[][][] cnt = new int[n][m][26];  // cnt[x][i][w]记录节点x和父节点之间的边权为w的个数
        int[] depth = new int[n];       // 记录n个节点的深度

        // 使用 dfs 从0节点开始 初始化pa、cnt 计算depth
        dfs(0, -1, g, pa, cnt, depth);

        // 计算 pa 和 cnt
        // 先枚举i,(也就是先算出所有节点的爷爷、再求所有节点爷爷的爷爷...
        for (int i = 0; i < m - 1; ++i) {   // 先枚举i,范围是0~m-2
            for (int x = 0; x < n; ++x) {   // 再枚举x
                int p = pa[x][i];           // 取出节点x的第2^i个父节点
                if (p != -1) {              
                    int pp = pa[p][i];      // 取出节点x的第2^i个父节点的第2^i个父节点
                    pa[x][i + 1] = pp;      // 赋值——x的第2^(i+1)个父节点
                    // 通过cnt[x][i]和cnt[p][i]计算 cnt[x][i+1]
                    for (int j = 0; j < 26; ++j) {
                        cnt[x][i + 1][j] = cnt[x][i][j] + cnt[p][i][j];
                    }
                }
            }
        }

        // 计算答案
        int[] ans = new int[queries.length];
        for (int qi = 0; qi < queries.length; qi++) {   // 枚举每一个查询
            int x = queries[qi][0], y = queries[qi][1];
            int pathLen = depth[x] + depth[y];          // x的深度和y的深度
            int[] cw = new int[26];                     // 统计各种边权在x和y之间出现的次数
            // 让 x 作为深度更小的那个节点
            if (depth[x] > depth[y]) {
                int t = x;
                x = y;
                y = t;
            }

            // 让 y 和 x 在同一深度(先让 y 跳 depth[y]-depth[x])
            for (int k = depth[y] - depth[x]; k > 0; k &= k - 1) {
                int i = Integer.numberOfTrailingZeros(k);
                int p = pa[y][i];
                for (int j = 0; j < 26; ++j) {
                    cw[j] += cnt[y][i][j];
                }
                y = p;
            }

            // y和x位于同一深度的时候可能位于同一个节点,那么就不用继续计算了
            if (y != x) {
                // 让 x 和 y 同时往上跳
                for (int i = m - 1; i >= 0; i--) {  // 从大到小尝试各种2^i跳法
                    int px = pa[x][i], py = pa[y][i];
                    // 如果px!=py,说明可以跳
                    if (px != py) {
                        for (int j = 0; j < 26; ++j) {
                            cw[j] += cnt[x][i][j] + cnt[y][i][j];
                        }   
                        x = px;
                        y = py;
                    }
                }
                // 因为跳到最后,x和y都是最近公共祖先的直系节点,所以px一定会=py
                // 手动计算cnt[j]
                for (int j = 0; j < 26; ++j) {
                    cw[j] += cnt[x][0][j] + cnt[y][0][j];
                }
                x = pa[x][0];   // x此时变成了 x 和 y 的最近公共祖先
            }

            int lca = x;
            pathLen -= depth[lca] * 2;
            int maxCw = 0;
            for (int i = 0; i < 26; ++i) maxCw = Math.max(maxCw, cw[i]);
            ans[qi] = pathLen - maxCw;
        }
        return ans;
    }

    public void dfs(int x, int fa, List<int[]>[] g, int[][] pa, int[][][] cnt, int[] depth) {
        pa[x][0] = fa;                  // 父节点
        for (int[] e: g[x]) {           // 枚举和x相连的每一条边
            int y = e[0], w = e[1];
            if (y != fa) {
                cnt[y][0][w] = 1;
                depth[y] = depth[x] + 1;
                dfs(y, x, g, pa, cnt, depth);
            }
        }
    }
}

相关题目

1483. 树节点的第 K 个祖先
2836. 在传球游戏中最大化函数值

成绩记录

在这里插入图片描述

喜报!应该要升 guardian 了!

在这里插入图片描述

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

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

相关文章

jQuery成功之路——jQuery动画效果和遍历效果概述

一、jQuery动画效果 1.1显示效果 方法 方法名称解释show([speed],[easing],[fn]])显示元素方法hide([speed],[easing],[fn]])隐藏元素方法toggle([speed],[easing],[fn])切换元素方法&#xff0c;显示的使之隐藏&#xff0c;隐藏的使之显示 参数 参数名称解释speed三种预…

CocosCreator3.8研究笔记(五)CocosCreator 脚本说明及使用(下)

在Cocos Creator中&#xff0c;脚本代码文件分为模块和插件两种方式&#xff1a; 模块一般就是项目的脚本&#xff0c;包含项目中创建的代码、引擎模块、第三方模块。 插件脚本&#xff0c;是指从 Cocos Creator 属性检查器中导入的插件&#xff0c;一般是引入第三方引入库文件…

管理类联考——数学——汇总篇——知识点突破——数据分析——排列组合

角度——&#x1f434; 角度——&#x1f469; 排列组合的基本步骤&#xff08;固定解题体系&#xff09; 先取后排&#xff1a;即先取出元素&#xff0c;后排列元素&#xff0c;切勿边取边排. 逐次进行&#xff1a;按照一定的顺序逐次进行排列组合. 实验结束&#xff1a;整个…

vscode保存格式化自动去掉分号、逗号、双引号

之前每次写完代码都是双引号还有分号&#xff0c;看着很难受&#xff0c;就像修改一下&#xff0c;让它变成单引号&#xff0c;并且不加上引号&#xff1a;如下形式&#xff0c;看着简洁清晰明了 修改方式&#xff1a;更改 settings.json 文件 快捷键“Ctrl Shift P”打开命令…

UmeTrack: Unified multi-view end-to-end hand tracking for VR 复现踩坑记录

在 github 上找到了开源代码&#xff1a;https://github.com/facebookresearch/UmeTrack/tree/main 环境配置 运行第三行&#xff0c;报错&#xff0c;缺少torch。改成先运行第四行&#xff0c;成功。 再运行第三行&#xff0c;报错&#xff0c;required to install pyproj…

uniapp 集成蓝牙打印功能(个人测试佳博打印机)

uniapp 集成蓝牙打印功能&#xff08;个人测试京博打印机&#xff09; uniapp 集成蓝牙打印功能集成佳博内置的接口 uniapp 集成蓝牙打印功能 大家好今天分析的是uniapp 集成蓝牙打印功能&#xff0c;个人开发是app,应该是支持H5(没试过) 集成佳博内置的接口 下载dome地址&…

空间复杂度和时间复杂度

&#x1f61c;作 者&#xff1a;是江迪呀✒️本文关键词&#xff1a;时间复杂度、空间复杂度、算法☀️每日 一言&#xff1a;车到山前必有路&#xff0c;船到码头自然直&#xff01; 一、前言 时间复杂度和空间复杂度是算法和数据结构领域中两个重要的概念&#…

服务器(I/O)之多路转接

五种IO模型 1、阻塞等待&#xff1a;在内核将数据准备好之前&#xff0c;系统调用会一直等待。所有的套接字&#xff0c;默认都是阻塞方式。 2、非阻塞等待&#xff1a;如果内核没有将数据准备好&#xff0c;系统调用仍然会返回&#xff0c;并且会返回EWUOLDBLOCK或者EAGAIN错…

springboot邮件发送和接收验证码

springboot邮件篇 要在Internet上提供电子邮件功能&#xff0c;必须有专门的电子邮件服务器。例如现在Internet很多提供邮件服务的厂商&#xff1a;新浪、搜狐、163、QQ邮箱等&#xff0c;他们都有自己的邮件服务器。这些服务器类似于现实生活中的邮局&#xff0c;它主要负责接…

【C语言】辗转相除法求最大公约数(详解)

辗转相除法求最大公约数 辗转相除法&#xff08;又称欧几里德算法&#xff09;是一种用于求解两个整数的最大公约数的方法。本文将使用C语言来实现辗转相除法&#xff0c;并对其原理进行解释。 辗转相除法的原理 辗转相除法的原理非常简单。假设有两个整数a和b&#xff0c;其…

Vue+Element-ui+SpringBoot搭建后端汽车租赁管理系统

最近在做项目&#xff0c;花了一周的时间搭建了一个十分完备的汽车租赁后端管理系统。页面采用纯Vue2Element-ui搭建&#xff0c;后端采用SpringbootMybatis搭建&#xff0c;数据库采用Mysql。包括了登录验证&#xff0c;根据不同权限进入不同界面、数据增删改查、表格分页、表…

iOS实时监控与报警器

在现代信息化社会中&#xff0c;即使我们不在电脑前面也能随时获取到最新的数据。而苹果公司提供的iOS推送通知功能为我们带来了一种全新的方式——通过手机接收实时监控和报警信息。 首先让我们了解一下iOS推送通知。它是一个强大且灵活可定制化程度高、适用于各类应用场景&a…

(二十一)大数据实战——Flume数据采集之复制和多路复用案例实战

前言 本节内容我们完成Flume数据采集的一个多路复用案例&#xff0c;使用三台服务器&#xff0c;一台服务器负责采集本地日志数据&#xff0c;通过使用Replicating ChannelSelector选择器&#xff0c;将采集到的数据分发到另外俩台服务器&#xff0c;一台服务器将数据存储到hd…

vue3 封装千分位分隔符自定义指令

toLocaleString作用&#xff1a;在没有指定区域的基本使用时&#xff0c;返回使用默认的语言环境和默认选项格式化的字符串。可点击进入MDN查看 // 千分位分隔符指令 import { Directive, DirectiveBinding } from vueconst thousandSeparator: Directive {mounted(el: any, …

window系统 bat脚本开启和关闭防火墙

前言 手动去关闭和开启防火墙太麻烦 命令 开始防火墙 netsh advfirewall set allprofiles state on关闭防火墙 netsh advfirewall set allprofiles state off

拦截器和异常处理器

拦截器和异常处理器 拦截器 拦截器(Interceptor)&#xff0c;主要完成请求参数的解析、将页面表单参数赋给值栈中相应属性、执行功能检验、程序异常调试等工作。 准备 创建模块 如下为完整的项目结构 web.xml <web-app xmlns"http://xmlns.jcp.org/xml/ns/javaee&qu…

【Linux】高级IO和多路转接 | select/poll/epoll

多路转接和高级IO 咳咳&#xff0c;写的时候出了点问题&#xff0c;标点符号全乱了&#xff08;批量替换了几次&#xff09;&#xff0c;干脆就把全文的逗号和句号都改成英文的了&#xff08;不然代码块里面的代码都是中文标点就跑不动了&#xff09; 1.高级IO 1.1 五种IO模型…

论文阅读《Nougat:Neural Optical Understanding for Academic Documents》

摘要 科学知识主要存储在书籍和科学期刊中&#xff0c;通常以PDF的形式。然而PDF格式会导致语义信息的损失&#xff0c;特别是对于数学表达式。我们提出了Nougat&#xff0c;这是一种视觉transformer模型&#xff0c;它执行OCR任务&#xff0c;用于将科学文档处理成标记语言&a…

【算法】分治法的基本思想和二分搜索的应用

创作不易&#xff0c;本篇文章如果帮助到了你&#xff0c;还请点赞 关注支持一下♡>&#x16966;<)!! 主页专栏有更多知识&#xff0c;如有疑问欢迎大家指正讨论&#xff0c;共同进步&#xff01; 更多算法分析与设计知识专栏&#xff1a;算法分析&#x1f525; 给大家跳…

手写Mybatis:第9章-细化XML语句构建器,完善静态SQL解析

文章目录 一、目标&#xff1a;XML语句构建器二、设计&#xff1a;XML语句构建器三、实现&#xff1a;XML语句构建器3.0 引入依赖3.1 工程结构3.2 XML语句构建器关系图3.3 I/O资源扫描3.4 SQL源码3.4.1 SQL对象3.4.2 SQL源码接口3.4.3 原始SQL源码实现类3.4.4 静态SQL源码实现类…