2020牛客寒假算法基础集训营3——G.牛牛的Link Power II【线段树】(画图详解)

本文主要是介绍2020牛客寒假算法基础集训营3——G.牛牛的Link Power II【线段树】(画图详解),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题目传送门


题目描述

牛牛有一颗大小为 n n n 的神奇 L i n k − C u t Link-Cut LinkCut 数组,数组上的每一个节点都有两种状态,一种为 l i n k link link 状态,另一种为cut状态。数组上任意一对处于link状态的无序点对(即(u,v)和(v,u)被认为是同一对)会产生dis(u,v)的link能量,dis(u,v)为数组上u到v的距离。

我们定义整个数组的Link能量为所有处于link状态的节点产生的link能量之和。

一开始数组上每个节点的状态将由一个长度大小为n的01串给出,'1’表示Link状态,'0’表示Cut状态。

牛牛想要知道一开始,以及每次操作之后整个数组的Link能量,为了避免这个数字过于庞大,你只用输出答案对 1 0 9 + 7 10^9+7 109+7取余后的结果即可。


输入描述:

第一行输入一个正整数 n ( 1 ≤ n ≤ 1 0 5 ) n(1 \leq n \leq 10^5) n(1n105)
接下里一行输入一个长度大小为n的01串表示数组的初始状态,'1’表示Link状态,'0’表示Cut状态。
接下来一行输入一个正整数 m ( 1 ≤ m ≤ 1 0 5 ) m(1 \leq m \leq 10^5) m(1m105)表示操作的数目

接下来m行,每行输入两个正整数 q , p o s ( q ∈ { 1 , 2 } , 1 ≤ p o s ≤ n ) q,pos(q \in \{ 1,2 \},1 \leq pos \leq n) q,pos(q{1,2},1posn)

  1. 当q=1时表示牛牛对数组的第pos个元素进行操作,将其赋值为1,保证在这个操作之前,该元素的值为0。
  2. 当q=2时表示牛牛对数组的第pos个元素进行操作,将其赋值为0,保证在这个操作之前,该元素的值为1。

输出描述:

请输出m+1行表示一开始,以及每次操作之后整个数组的Link能量,为了避免这个数字过于庞大,你只用输出答案对 1 0 9 + 7 10^9+7 109+7取余后的结果即可。


输入

5
00001
7
1 1
2 5
2 1
1 2
1 4
1 3
1 1


输出

0
4
0
0
0
2
4
10


题意

  • 给你一个只含 0 0 0 1 1 1的串,定义串的Link值为串中两个的 1 1 1之间的距离的和, ( u , v ) (u,v) (u,v) ( v , u ) (v,u) (v,u)被看认为是同一对,有 m m m次操作,每次操作可以把串中某个 1 1 1变为 0 0 0,或者把某个 0 0 0变为 1 1 1,求一开始和每次操作后串的 L i n k Link Link值。

题解

  • 用线段树直接维护即可
  • 区间内 1 1 1 的数量 c n t cnt cnt L i n k Link Link 值,区间内所有的 1 1 1 到区间左边界的距离之和 d l dl dl,区间内所有的 1 1 1 的区间右边界距离之和 d r dr dr
  • 查询的时候只要输出 t r e e [ 1 ] . L i n k tree[1].Link tree[1].Link 即可,修改时只需改变区间内 c n t cnt cnt,然后合并区间
  • 现在我们考虑合并区间
    在这里插入图片描述
    如图,考虑这样的普遍情况,我们有左区间、右区间,现在要将两区间合并。
    合并之后的 L i n k Link Link = = = 左 区 间 内 部 的 L i n k + 右 区 间 内 部 的 L i n k + 跨 越 中 间 m i d ( 黄 线 ) 的 贡 献 ( 紫 线 ) 左区间内部的Link+右区间内部的Link+跨越中间mid(黄线)的贡献(紫线) Link+Link+mid线(线)
    那么我们只需要得到 紫 线 紫线 线 就可以求得合并的 L i n k Link Link
    我们将 紫 线 紫线 线 分成左右两部分来看,很容易发现, 紫 线 左 部 分 紫线左部分 线 = 右 . c n t ∗ 左 . d r =右.cnt*左.dr =.cnt.dr,同理, 紫 线 右 部 分 紫线右部分 线 = 左 . c n t ∗ 右 . d l =左.cnt*右.dl =.cnt.dl
  • 但是这样并不是最终答案,考虑下图
    在这里插入图片描述
    这是数据 11111 11111 11111 的图,经过我们的分析,显然上述合并方程是正确的,但是忽略了一个问题,就是 d l dl dl d r dr dr 的取值问题。
    我们定义 d l : 区 间 内 所 有 的 1 到 区 间 左 端 点 的 距 离 和 dl:区间内所有的1到区间左端点的距离和 dl:1,但是对于左端点如果是1的时候,这个距离是没有被计算到的(如上图)
    当我们合并的时候,正常来说 右 . d l = 1 + 2 = 3 右.dl=1+2=3 .dl=1+2=3,但实际上,第四个1到左端点的距离是0,我们得到的 右 . d l = 2 右.dl=2 .dl=2
    那我们发现,缺少的就是蓝色框中的长度,容易发现,这部分是 左 边 所 有 1 左边所有1 1 右 边 所 有 1 右边所有1 1 共同贡献的,而整个蓝框的长度就是1个单位长度,那么缺少的是 条 数 ∗ 1 = 条 数 条数*1=条数 1=。那么我们就可以得到 缺 少 的 部 分 = 左 . c n t ∗ 右 . c n t ∗ 1 缺少的部分 = 左.cnt*右.cnt*1 =.cnt.cnt1,计算 合 并 L i n k 合并Link Link 的时候加上这部分即可。
  • 至此,这道题就算解决完了,注意取模,不要爆精度即可。

AC-Code

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxn = 1e5 + 7;
const int mod = 1e9 + 7;string s;
struct Node {int left, right;ll Link, dl, dr, cnt;
};
struct Segment_Tree {
#define mid ((tree[rt].left + tree[rt].right) >> 1)Node tree[maxn << 2];void PushUp(int rt) {tree[rt].cnt = tree[rt << 1].cnt + tree[rt << 1 | 1].cnt;int t = (tree[rt << 1].cnt * tree[rt << 1 | 1].dl + tree[rt << 1 | 1].cnt * tree[rt << 1].dr) % mod; // 跨越两个区间mid的贡献tree[rt].Link = tree[rt << 1].Link + tree[rt << 1 | 1].Link + t + tree[rt << 1].cnt * tree[rt << 1 | 1].cnt;tree[rt].dl = tree[rt << 1].dl + tree[rt << 1 | 1].dl + tree[rt << 1 | 1].cnt * (tree[rt << 1].right - tree[rt << 1].left + 1) % mod;tree[rt].dr = tree[rt << 1].dr + tree[rt << 1 | 1].dr + tree[rt << 1].cnt * (tree[rt << 1 | 1].right - tree[rt << 1 | 1].left + 1) % mod;}void build(int rt, int l, int r) {tree[rt].left = l;tree[rt].right = r;if (l == r) {tree[rt].dl = tree[rt].dr = tree[rt].Link = 0;if (s[l] == '1')	tree[rt].cnt = 1;else 	tree[rt].cnt = 0;return;}build(rt << 1, l, mid);build(rt << 1 | 1, mid + 1, r);PushUp(rt);}void update(int rt, int pos, int val) {if (tree[rt].left == tree[rt].right) {tree[rt].cnt = val;return;}if (pos <= mid)	update(rt << 1, pos, val);else	update(rt << 1 | 1, pos, val);PushUp(rt);}
#undef mid
};
Segment_Tree st;
int main() {int n;	while (cin >> n) {cin >> s;	s = " " + s;st.build(1, 1, n);cout << st.tree[1].Link % mod << endl;int m;	cin >> m;while (m--) {int q, pos;	cin >> q >> pos;st.update(1, pos, q == 1 ? 1 : 0);cout << st.tree[1].Link % mod << endl;}}
}

这篇关于2020牛客寒假算法基础集训营3——G.牛牛的Link Power II【线段树】(画图详解)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/695239

相关文章

RedHat运维-Linux文本操作基础-AWK进阶

你不用整理,跟着敲一遍,有个印象,然后把它保存到本地,以后要用再去看,如果有了新东西,你自个再添加。这是我参考牛客上的shell编程专项题,只不过换成了问答的方式而已。不用背,就算是我自己亲自敲,我现在好多也记不住。 1. 输出nowcoder.txt文件第5行的内容 2. 输出nowcoder.txt文件第6行的内容 3. 输出nowcoder.txt文件第7行的内容 4. 输出nowcode

Vim使用基础篇

本文内容大部分来自 vimtutor,自带的教程的总结。在终端输入vimtutor 即可进入教程。 先总结一下,然后再分别介绍正常模式,插入模式,和可视模式三种模式下的命令。 目录 看完以后的汇总 1.正常模式(Normal模式) 1.移动光标 2.删除 3.【:】输入符 4.撤销 5.替换 6.重复命令【. ; ,】 7.复制粘贴 8.缩进 2.插入模式 INSERT

零基础STM32单片机编程入门(一)初识STM32单片机

文章目录 一.概要二.单片机型号命名规则三.STM32F103系统架构四.STM32F103C8T6单片机启动流程五.STM32F103C8T6单片机主要外设资源六.编程过程中芯片数据手册的作用1.单片机外设资源情况2.STM32单片机内部框图3.STM32单片机管脚图4.STM32单片机每个管脚可配功能5.单片机功耗数据6.FALSH编程时间,擦写次数7.I/O高低电平电压表格8.外设接口

十四、观察者模式与访问者模式详解

21.观察者模式 21.1.课程目标 1、 掌握观察者模式和访问者模式的应用场景。 2、 掌握观察者模式在具体业务场景中的应用。 3、 了解访问者模式的双分派。 4、 观察者模式和访问者模式的优、缺点。 21.2.内容定位 1、 有 Swing开发经验的人群更容易理解观察者模式。 2、 访问者模式被称为最复杂的设计模式。 21.3.观察者模式 观 察 者 模 式 ( Obser

【操作系统】信号Signal超详解|捕捉函数

🔥博客主页: 我要成为C++领域大神🎥系列专栏:【C++核心编程】 【计算机网络】 【Linux编程】 【操作系统】 ❤️感谢大家点赞👍收藏⭐评论✍️ 本博客致力于知识分享,与更多的人进行学习交流 ​ 如何触发信号 信号是Linux下的经典技术,一般操作系统利用信号杀死违规进程,典型进程干预手段,信号除了杀死进程外也可以挂起进程 kill -l 查看系统支持的信号

ps基础入门

1.基础      1.1新建文件      1.2创建指定形状      1.4移动工具          1.41移动画布中的任意元素          1.42移动画布          1.43修改画布大小          1.44修改图像大小      1.5框选工具      1.6矩形工具      1.7图层          1.71图层颜色修改          1

Jitter Injection详解

一、定义与作用 Jitter Injection,即抖动注入,是一种在通信系统中人为地添加抖动的技术。该技术通过在发送端对数据包进行延迟和抖动调整,以实现对整个通信系统的时延和抖动的控制。其主要作用包括: 改善传输质量:通过调整数据包的时延和抖动,可以有效地降低误码率,提高数据传输的可靠性。均衡网络负载:通过对不同的数据流进行不同程度的抖动注入,可以实现网络资源的合理分配,提高整体传输效率。增

代码随想录算法训练营:12/60

非科班学习算法day12 | LeetCode150:逆波兰表达式 ,Leetcode239: 滑动窗口最大值  目录 介绍 一、基础概念补充: 1.c++字符串转为数字 1. std::stoi, std::stol, std::stoll, std::stoul, std::stoull(最常用) 2. std::stringstream 3. std::atoi, std

人工智能机器学习算法总结神经网络算法(前向及反向传播)

1.定义,意义和优缺点 定义: 神经网络算法是一种模仿人类大脑神经元之间连接方式的机器学习算法。通过多层神经元的组合和激活函数的非线性转换,神经网络能够学习数据的特征和模式,实现对复杂数据的建模和预测。(我们可以借助人类的神经元模型来更好的帮助我们理解该算法的本质,不过这里需要说明的是,虽然名字是神经网络,并且结构等等也是借鉴了神经网络,但其原型以及算法本质上还和生物层面的神经网络运行原理存在

Steam邮件推送内容有哪些?配置教程详解!

Steam邮件推送功能是否安全?如何个性化邮件推送内容? Steam作为全球最大的数字游戏分发平台之一,不仅提供了海量的游戏资源,还通过邮件推送为用户提供最新的游戏信息、促销活动和个性化推荐。AokSend将详细介绍Steam邮件推送的主要内容。 Steam邮件推送:促销优惠 每当平台举办大型促销活动,如夏季促销、冬季促销、黑色星期五等,用户都会收到邮件通知。这些邮件详细列出了打折游戏、