AtCoder Beginner Contest 357-F
Problem
同步于博客
Problem
You are given sequences of length \(N\), \(A=(A_1,A_2,\ldots,A_N)\) and \(B=(B_1,B_2,\ldots,B_N)\).
You are also given \(Q\) queries to process in order.
There are three types of queries:
1 l r x: Add \(x\) to each of \(A_l, A_{l+1}, \ldots, A_r\).2 l r x: Add \(x\) to each of \(B_l, B_{l+1}, \ldots, B_r\).3 l r: Print the remainder of \(\displaystyle\sum_{i=l}^r (A_i\times B_i)\) when divided by \(998244353\).
给你两个长度为 \(N\) 的序列 \(A=(A_1,A_2,\ldots,A_N)\) 和 \(B=(B_1,B_2,\ldots,B_N)\) 的序列。
需要按顺序处理 \(Q\) 个查询。
查询有三种类型:
1 l r x: 在 \(A_l, A_{l+1}, \ldots, A_r\) 中的每一条添加 \(x\) 。2 l r x: 向 \(B_l, B_{l+1}, \ldots, B_r\) 中的每一条添加 \(x\) 。3 l r: 打印 \(\displaystyle\sum_{i=l}^r (A_i\times B_i)\) 除以 \(998244353\) 的余数。
Constraints
- \(1\leq N,Q\leq 2\times 10^5\)
- \(0\leq A_i,B_i\leq 10^9\)
- \(1\leq l\leq r\leq N\)
- \(1\leq x\leq 10^9\)
- 全是整数
Solution
看到 \(N,Q\) 的取值范围,发现需要在 \(n\log n\) 的复杂度内解决。值域 \(1\le x\le 10^9\) 也不好做文章。
数列\(A,B\)在\([l,r]\)上在进行操作1 l r x和2 l r y后会变为
&\sum_{i=l}^r(A_i+x)\times (B_i+y)\\
&=\sum_{i=l}^rA_i\times B_i+x\sum_{i=l}^r B_i+y\sum_{i=l}^r A_i+xy(r-l+1)
\end{align}
\]
故我们使用线段树维护三个量 \(\sum A_i\times B_i,\sum A_i,\sum B_i\) 和lazy标记 \(Add_a,Add_b\)
当执行1 l r x 时
- \(\sum A_i\times B_i\) 增加 \(x\sum B_i+x\cdot Add_b\cdot(r-l+1)\)
- \(\sum A_i\) 增加 \(x(r-l+1)\)
- \(Add_a\) 增加 \(x\)
当执行2 l r y 时
- \(\sum A_i\times B_i\) 增加 \(y\sum A_i+y\cdot Add_a\cdot(r-l+1)\)
- \(\sum B_i\) 增加 \(y(r-l+1)\)
- \(Add_b\) 增加 \(y\)
Code
#define P 998244353ll
#define N 800010
struct Edge{
LL l,r,sumA,sumB,sum,addA,addB;
#define l(x) tree[x].l
#define r(x) tree[x].r
#define sumA(x) tree[x].sumA
#define sumB(x) tree[x].sumB
#define sum(x) tree[x].sum
#define addA(x) tree[x].addA
#define addB(x) tree[x].addB
}tree[N];
LL n,m;
LL A[N],B[N];
void build(LL l,LL r,LL p)
{
l(p)=l,r(p)=r;
if(l==r){sumA(p)=A[l]%P;sumB(p)=B[l]%P;sum(p)=sumA(p)*sumB(p)%P;return;}
LL mid=(l+r)>>1;
build(l,mid,p<<1);
build(mid+1,r,p<<1|1);
sum(p)=sum(p<<1)+sum(p<<1|1);
sumA(p)=sumA(p<<1)+sumA(p<<1|1);
sumB(p)=sumB(p<<1)+sumB(p<<1|1);
sum(p)%=P;
sumA(p)%=P;
sumB(p)%=P;
}
void spread(LL p)
{
if(addA(p)||addB(p))
{
addA(p<<1)+=addA(p);
addA(p<<1|1)+=addA(p);
addB(p<<1)+=addB(p);
addB(p<<1|1)+=addB(p);
sum(p<<1)+=addA(p)*sumB(p<<1)+addB(p)*sumA(p<<1)+addA(p)*addB(p)%P*(r(p<<1)-l(p<<1)+1);
sum(p<<1|1)+=addA(p)*sumB(p<<1|1)+addB(p)*sumA(p<<1|1)+addA(p)*addB(p)%P*(r(p<<1|1)-l(p<<1|1)+1);
sumA(p<<1)+=addA(p)*(r(p<<1)-l(p<<1)+1);
sumB(p<<1)+=addB(p)*(r(p<<1)-l(p<<1)+1);
sumA(p<<1|1)+=addA(p)*(r(p<<1|1)-l(p<<1|1)+1);
sumB(p<<1|1)+=addB(p)*(r(p<<1|1)-l(p<<1|1)+1);
addA(p)=addB(p)=0;
sum(p<<1)%=P;
sum(p<<1|1)%=P;
sumA(p<<1)%=P;
sumA(p<<1|1)%=P;
sumB(p<<1)%=P;
sumB(p<<1|1)%=P;
addA(p<<1)%=P;
addB(p<<1)%=P;
addA(p<<1|1)%=P;
addB(p<<1|1)%=P;
}
}
void change(LL l,LL r,LL p,LL k,bool isA)
{
if(l<=l(p)&&r>=r(p)){
if(isA) addA(p)+=k,sum(p)+=k*sumB(p),sumA(p)+=k*(r(p)-l(p)+1);
else addB(p)+=k,sum(p)+=k*sumA(p),sumB(p)+=k*(r(p)-l(p)+1);
sumA(p)%=P;sumB(p)%=P;sum(p)%=P;
addA(p)%=P;addB(p)%=P;
return;
}
LL mid=l(p)+r(p)>>1;
spread(p);
if(l<=mid) change(l,r,p<<1,k,isA);
if(r>mid) change(l,r,p<<1|1,k,isA);
sum(p)=sum(p<<1)+sum(p<<1|1);
sumA(p)=sumA(p<<1)+sumA(p<<1|1);
sumB(p)=sumB(p<<1)+sumB(p<<1|1);
sum(p)%=P;
sumA(p)%=P;
sumB(p)%=P;
}
LL ask(LL l,LL r,LL p)
{
if(l<=l(p)&&r>=r(p)) return sum(p);
spread(p);
LL mid=l(p)+r(p)>>1,val=0;
if(l<=mid) val+=ask(l,r,p<<1);
val%=P;
if(r>mid) val+=ask(l,r,p<<1|1);
val%=P;
return val;
}
int main()
{
ios::sync_with_stdio(false);
cin.tie(0);
cout.tie(0);
cout.precision(10);
int q;
cin>>n>>q;
for(int i=1;i<=n;i++) cin>>A[i];
for(int i=1;i<=n;i++) cin>>B[i];
build(1,n,1);
while(q--)
{
int op,l,r,x;
cin>>op>>l>>r;
if(op!=3) cin>>x,change(l,r,1,x,(op==1));
else cout<<ask(l,r,1)<<endl;
}
return 0;
}
Attention
开LL,多取模。
AtCoder Beginner Contest 357-F的更多相关文章
- AtCoder Beginner Contest 137 F
AtCoder Beginner Contest 137 F 数论鬼题(虽然不算特别数论) 希望你在浏览这篇题解前已经知道了费马小定理 利用用费马小定理构造函数\(g(x)=(x-i)^{P-1}\) ...
- AtCoder Beginner Contest 261 F // 树状数组
题目链接:F - Sorting Color Balls (atcoder.jp) 题意: 有n个球,球有颜色和数字.对相邻的两球进行交换时,若颜色不同,需要花费1的代价.求将球排成数字不降的顺序,所 ...
- AtCoder Beginner Contest 260 F - Find 4-cycle
题目传送门:F - Find 4-cycle (atcoder.jp) 题意: 给定一个无向图,其包含了S.T两个独立点集(即S.T内部间的任意两点之间不存在边),再给出图中的M条边(S中的点与T中的 ...
- AtCoder Beginner Contest 253 F - Operations on a Matrix // 树状数组
题目传送门:F - Operations on a Matrix (atcoder.jp) 题意: 给一个N*M大小的零矩阵,以及Q次操作.操作1(l,r,x):对于 [l,r] 区间内的每列都加上x ...
- AtCoder Beginner Contest 249 F - Ignore Operations // 贪心 + 大根堆
传送门:F - Keep Connect (atcoder.jp) 题意: 给定长度为N的操作(ti,yi). 给定初值为0的x,对其进行操作:当t为1时,将x替换为y:当t为2时,将x加上y. 最多 ...
- AtCoder Beginner Contest 247 F - Cards // dp + 并查集
原题链接:F - Cards (atcoder.jp) 题意: 给定N张牌,每张牌正反面各有一个数,所有牌的正面.反面分别构成大小为N的排列P,Q. 求有多少种摆放方式,使得N张牌朝上的数字构成一个1 ...
- AtCoder Beginner Contest 133 F Colorful Tree
Colorful Tree 思路: 如果强制在线的化可以用树链剖分. 但这道题不强制在线,那么就可以将询问进行差分,最后dfs时再计算每个答案的修改值, 只要维护两个数组就可以了,分别表示根节点到当前 ...
- AtCoder Beginner Contest 171-175 F
171 F - Strivore 直接把初始字符当成隔板,统计的方案数会有重复 为了避免重复情况,规定隔板字母尽可能最后出现,即在隔板字母后面不能插入含隔板字母的字符串 所以在隔板字母后插入的字符只有 ...
- AtCoder Beginner Contest 182 F
F - Valid payments 简化题意:有\(n\)种面值的货币,保证\(a[1]=1,且a[i+1]是a[i]的倍数\). 有一个价格为\(x\)元的商品,付款\(y\)元,找零\(y-x\ ...
- AtCoder Beginner Contest 215 F题题解
F - Dist Max 2 什么时候我才能突破\(F\)题的大关... 算了,不说了,看题. 简化题意:给定\(n\)个点的坐标,定义没两个点的距离为\(min(|x_i-x_j|,|y_i-y_j ...
随机推荐
- pytest 框架使用规则
使用pytest 注意 有时候常常会调用时文件未执行,就是没注意命名规范 如何调用-命令行调用 pytest 批量运行测试用例 单个用例调试成功后,接下来我们要进行所有脚本统一执行 我要执行testc ...
- kubernetes failed to create kubelet: misconfiguration: kubelet cgroup driver: "cgroupfs" is different from docker cgroup driver: "systemd"
错误原因 kubernetes 的文件驱动与 docker 不一致,导致镜像无法启动. docker info 可以看到驱动方式 Cgroup Driver: systemd. 解决方案 统一资源管理 ...
- linux 上安装portainer.io
inux 上安装portainer.io 1.portainer.io是什么? 2.安装,运行镜像 3.登陆1.portainer.io是什么?Portainer是一个轻量级的Doc ...
- 使用 vscode-jest 插件
vscode-jest [error] Abort jest session: Not able to auto detect a valid jest command: multiple candi ...
- 包和抽象类介绍--java进阶day02
1.package包 导包第二点需要注意 a包和b包都存有Student类,c包存有测试类,我们在c中创建Student对象,系统会询问你要哪个包的Student类,并自动帮你导包 . 在导完a包的学 ...
- Linux下时区/系统时间/硬件时间的设置
涨姿势,顺带笔记,留爪. 先简述下时区/系统时间/硬件时间的3个主要命令吧 tzselect tzselect命令主要针对时区设置和查看 tz=timezone的缩写,直译=时区 date date命 ...
- 掌握FastAPI与Pydantic的跨字段验证技巧
title: 掌握FastAPI与Pydantic的跨字段验证技巧 date: 2025/04/01 00:32:07 updated: 2025/04/01 00:32:07 author: cmd ...
- 聊聊AI Agent与AI 数字分身
提供AI应用咨询+陪跑服务,有需要回复1 Manus爆火后,网上出现了很多AI热门名词,比如Agent.AI分身,并且有一张技术架构实现图: 怎么说呢,也许这张图是对的,但就我这边实际的项目实践情况以 ...
- Cloud Native CI/CD: tekton and argocd
https://platform9.com/blog/argo-cd-vs-tekton-vs-jenkins-x-finding-the-right-gitops-tooling/ implemen ...
- TM1637读取键值调试笔记
因为项目原因需要用到TM1637,实现驱动数码管和按键扫描,参考了网络上搜索到的一些例程,基本实现了功能要求,能够实现数码管点亮和按键扫描. 调试过程中也出现一些问题,现在描述一下问题和解决方 ...