[蓝桥杯学习]树上差分

差分

前缀和 sum_i = sum_i-1 + a_i

差分 diff_i = a_i - a_i-1

差分的好处

点的差分

问题引入
解决问题

要用到差分的思想,每次从叶子向上的回溯,让父结点+=子结点的cnt值,但是仅仅这样,还不行

回溯的过程中,LCA被加了两次,要减去一,LCA的父结点原本应该没有数值的,但是因为会加上LCA 的值,所以会多1,要减去。

此上操作,便实现了,只是把s t 两个结点的值赋一,便将s到t路径上的点经过次数赋值为1

代码实现

cnt计算的代码实现,如下

cpp 复制代码
void dfs1(int x){
    for(int i=head[x];i;i=edge[i].nex){
        int u=edge[i].to;
        if(u!=fa[x][0]){
            dfs1(u);
            cnt[x]+=cnt[u];
        }
    }
    return ;
}

cnt[x]+=cnt[u] ,在dfs1 之后,就是从子结点回到父结点后进行cnt相加。

边的差分

问题引入

将路上每条边都权值加z

解决问题

把边塞给点,边的权值放在点上,使用差分的方法,每次修改都是O(1)的时间复杂度,将最后的结果最坏下是O(n)遍历要求的区间,就可以得到区间权值和。

解释为什么要把 dlt[lca(u,v)]-=2x

因为根据方法,回溯时父结点会加上子结点的权值,所以LCA会加上左右两边的孩子的权值(即边的权值),根据我们给点权值的定义,意思是,会经过该边两次,但是该边没有经过两次,所以会减去2x

代码实现

与点上的差分一样的

时间复杂度分析

维护的时间复杂度是O(1),最后dfs一遍统计答案,最坏时间复杂度是O(n)

相关推荐
立志成为大牛的小牛18 小时前
数据结构——五十一、散列表的基本概念(王道408)
开发语言·数据结构·学习·程序人生·算法·散列表
闲猿类21 小时前
嵌入式第九天学习
linux·c语言·学习·算法·嵌入式
专注于大数据技术栈21 小时前
java学习--main方法
java·开发语言·学习
d111111111d1 天前
关于STM32的选项字节的问题:如果我通过操作指针把数据写在了单片机的选项字节区域那么换别的程序时候数据会进行变化吗?
笔记·stm32·单片机·嵌入式硬件·学习
安如衫1 天前
【机器学习基础】Attention in Transformers:注意力机制
笔记·深度学习·学习·机器学习·注意力机制
十安_数学好题速析1 天前
幂次之争:巧用对称性破解指数不等式
笔记·学习·高考
一 乐1 天前
运动会|基于SpingBoot+vue的高校体育运动会管理系统(源码+数据库+文档)
java·前端·javascript·数据库·vue.js·学习·springboot
@曾记否1 天前
【Betaflight源码学习】Betaflight 嵌入式操作系统架构解析:与 FreeRTOS 的深度对比
学习·架构
我命由我123451 天前
Excel - Excel 找回意外关闭的未保存的文档
学习·职场和发展·excel·求职招聘·职场发展·运维开发·学习方法
落羽的落羽1 天前
【Linux系统】解明进程优先级与切换调度O(1)算法
linux·服务器·c++·人工智能·学习·算法·机器学习