Luogu3373【模板】线段树2

时间:2022-09-04 11:34:27

P3373【模板】线段树2

题目描述

如题,已知一个数列,你需要进行下面两种操作:

1.将某区间每一个数加上x

2.将某区间每一个数乘上x

3.求出某区间每一个数的和

输入输出格式

输入格式:

第一行包含三个整数N、M、P,分别表示该数列数字的个数、操作的总个数和模数。

第二行包含N个用空格分隔的整数,其中第i个数字表示数列第i项的初始值。

接下来M行每行包含3或4个整数,表示一个操作,具体如下:

操作1: 格式:1 x y k 含义:将区间[x,y]内每个数乘上k

操作2: 格式:2 x y k 含义:将区间[x,y]内每个数加上k

操作3: 格式:3 x y 含义:输出区间[x,y]内每个数的和对P取模所得的结果

输出格式:

输出包含若干行整数,即为所有操作3的结果。

输入输出样例

输入样例#1:
5 5 38
1 5 4 2 3
2 1 4 1
3 2 5
1 2 4 2
2 3 5 5
3 1 4
输出样例#1:
17
2

说明

时空限制:1000ms,128M

数据规模:

对于30%的数据:N<=8,M<=10

对于70%的数据:N<=1000,M<=10000

对于100%的数据:N<=100000,M<=100000

(数据已经过加强^_^)

样例说明:

Luogu3373【模板】线段树2

故输出应为17、2(40 mod 38=2)

线段树多个lazy标记的应用。

每个点上的标记表示自己儿子要乘或加上多少。

标记下放时,儿子节点的add标记与儿子节点的值分别先乘上父亲节点的乘标记再加上父亲节点的加标记(因为父亲节点的加标记已经乘上了一些乘标记了,不需要再乘一次)

儿子节点的乘标记直接乘上父亲节点的乘标记。

 #include<iostream>
 #include<cstdio>
 using namespace std;
 ;
 typedef long long ll;
 struct node{
     int l,r;
     ll val,add,time;
 }T[N*];
 ll Mod,a[N];
 inline void build(int l,int r,int x)
 {
     T[x].l=l;
     T[x].r=r;
     T[x].add=;
     T[x].time=;
     if(l!=r)
     {
         ;
         build(l,m,x*);
         build(m+,r,x*+);
         T[x].val=(T[x*].val+T[x*+].val)%Mod;
     }
     else T[x].val=a[l]%Mod;
 }
 inline ll cnt(int l,int r,int x)
 {
     if(l==T[x].l&&r==T[x].r) return T[x].val;
     ;i<=;i++)
     {
         T[x*+i].val=(T[x*+i].val*T[x].time+T[x].add*(T[x*+i].r-T[x*+i].l+))%Mod;
         T[x*+i].add=(T[x*+i].add*T[x].time+T[x].add)%Mod;
         T[x*+i].time=(T[x].time*T[x*+i].time)%Mod;
     }
     T[x].add=;
     T[x].time=;
     ll res=;
     ;
     );
     +);
     )+cnt(m+,r,x*+))%Mod;
     return res;
 }
 inline void addc(int l,int r,int x,ll c,int t)
 {
     if(l==T[x].l&&r==T[x].r)
     {
         )
         {
             T[x].time=(T[x].time*c)%Mod;
             T[x].val=(T[x].val*c)%Mod;
             T[x].add=(T[x].add*c)%Mod;
         }
         else
         {
             T[x].val=(T[x].val+c*(r-l+))%Mod;
             T[x].add=(T[x].add+c)%Mod;
         }
         return;
     }
     ;i<=;i++)
     {
         T[x*+i].val=(T[x*+i].val*T[x].time+T[x].add*(T[x*+i].r-T[x*+i].l+))%Mod;
         T[x*+i].add=(T[x*+i].add*T[x].time+T[x].add)%Mod;
         T[x*+i].time=(T[x].time*T[x*+i].time)%Mod;
     }
     T[x].add=;
     T[x].time=;
     ;
     ,c,t);
     +,c,t);
     else
     {
         addc(l,m,x*,c,t);
         addc(m+,r,x*+,c,t);
     }
     T[x].val=(T[x*].val+T[x*+].val)%Mod;
 }
 int main()
 {
     int n,m,t,l,r;
     ll k;
     scanf("%d%d%lld",&n,&m,&Mod);
     ;i<=n;i++) scanf("%lld",&a[i]);
     build(,n,);
     ;i<=m;i++)
     {
         scanf("%d%d%d",&t,&l,&r);
         ||t==)
         {
             scanf("%lld",&k);
             addc(l,r,,k,t);
         }
         ));
     }
     ;
 }

Luogu3373【模板】线段树2的更多相关文章

  1. hdu 1754 I Hate It &lpar;模板线段树&rpar;

    http://acm.hdu.edu.cn/showproblem.php?pid=1754 I Hate It Time Limit: 9000/3000 MS (Java/Others)    M ...

  2. 洛谷P3373 &lbrack;模板&rsqb;线段树 2&lpar;区间增减&period;乘 区间求和&rpar;

    To 洛谷.3373 [模板]线段树2 题目描述 如题,已知一个数列,你需要进行下面两种操作: 1.将某区间每一个数加上x 2.将某区间每一个数乘上x 3.求出某区间每一个数的和 输入输出格式 输入格 ...

  3. hdu3966 树链剖分点权模板&plus;线段树区间更新&sol;树状数组区间更新单点查询

    点权树的模板题,另外发现树状数组也是可以区间更新的.. 注意在对链进行操作时方向不要搞错 线段树版本 #include<bits/stdc++.h> using namespace std ...

  4. 【线段树】【P3372】模板-线段树

    百度百科 Definition&Solution 线段树是一种log级别的树形结构,可以处理区间修改以及区间查询问题.期望情况下,复杂度为O(nlogn). 核心思想见百度百科,线段树即将每个 ...

  5. 算法模板——线段树6(二维线段树:区域加法&plus;区域求和)(求助phile)

    实现功能——对于一个N×M的方格,1:输入一个区域,将此区域全部值作加法:2:输入一个区域,求此区域全部值的和 其实和一维线段树同理,只是不知道为什么速度比想象的慢那么多,求解释...@acphile ...

  6. 【洛谷 p3373】模板-线段树 2(数据结构--线段树)

    题意:已知一个数列,你需要进行下面三种操作:1.将某区间每一个数加上x:2.将某区间每一个数乘上x:3.求出某区间每一个数的和. 解法:(唉 :-(,这题卡住我了......)对于加法和乘法的混合操作 ...

  7. 【洛谷 p3372】模板-线段树 1(数据结构--线段树)

    题目:已知一个数列,你需要进行下面两种操作:1.将某区间每一个数加上x:2.求出某区间每一个数的和. 解法:如题,模版题.需要加上 lazy 标记,也就是我的 upd.lazy 标记的思路就是对一个结 ...

  8. hdu 1754 I Hate It &lpar;线段树、单点更新&rpar;(PS:ios&colon;&colon;sync&lowbar;with&lowbar;stdio&lpar;false&rpar;可以加快cin、cout的读取写出速度)

    I Hate ItTime Limit: 9000/3000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Su ...

  9. poj 3468 A Simple Problem with Integers 线段树 题解《挑战程序设计竞赛》

    地址 http://poj.org/problem?id=3468 线段树模板 要背下此模板 线段树 #include <iostream> #include <vector> ...

  10. 线段树 扫描线 L - Atlantis HDU - 1542 M - City Horizon POJ - 3277 N - Paint the Wall HDU - 1543

    学习博客推荐——线段树+扫描线(有关扫描线的理解) 我觉得要注意的几点 1 我的模板线段树的叶子节点存的都是 x[L]~x[L+1] 2 如果没有必要这个lazy 标志是可以不下传的 也就省了一个pu ...

随机推荐

  1. centos7 mysql数据库安装和配置

    一.系统环境 yum update升级以后的系统版本为 [root@yl-web yl]# cat /etc/redhat-release CentOS Linux release 7.1.1503 ...

  2. Spring IoC源码解决——工具篇Eclipse

    题外话 对于Spring框架,平时都是点到为止,停留在会用的程度.一直以来都想深入学习下,刚好最近看到<Spring源码深度解析>,所以想随着书本深入学习一下. 如果用Maven 如果使用 ...

  3. PHP遍历目录四种方法

    学习SPL的时候,遇到了DirectoryIterator这个目录类,谢了一下遍历目录的方法.于是总结一下遍历目录的四种写法 如下: <?php /* * 方法一:利用SPL的目录类,这个很简单 ...

  4. Threadpool dump

    Microsoft (R) Windows Debugger Version 10.0.10586.567 AMD64Copyright (c) Microsoft Corporation. All ...

  5. 【原创】回溯线搜索 Backtracking line search

    机器学习中很多数值优化算法都会用到线搜索(line search).线搜索的目的是在搜索方向上找到是目标函数\(f(x)\)最小的点.然而,精确找到最小点比较耗时,由于搜索方向本来就是近似,所以用较小 ...

  6. 使用buildbot实现持续集成(转载)

    转载自:http://www.oschina.net/p/buildbot 使用 Buildot 实现持续集成 使用基于 Python 的工具实现持续集成的理论与实践 牛仔式编码的日子在大多数组织中早 ...

  7. Spring AOP &plus; AspectJ annotation example

    In this tutorial, we show you how to integrate AspectJ annotation with Spring AOP framework. In simp ...

  8. 数论&sol;the first wave

    线性筛素数(原来我之前学的不是线性的啊... void getprime(){ rep(i,2,nmax){ if(!vis[i]) prime[++prime[0]]=i; for(int j=1; ...

  9. delphi 7 信息对话框的按钮屏蔽键盘操作&comma;只允许鼠标点击

    本问题由 delphi 学友QQ群中一位群友提出,个人觉得是一个好问题. 本教程源码下载 本教程面向新手,希望大家能从中学到除了本功能之外的真正编程技能. 本功能的实现原理,用窗口当对话框的界面,在界 ...

  10. JDK源码学习系列01----String

                                                     JDK源码学习系列01----String 写在最前面: 这是我JDK源码学习系列的第一篇博文,我知道 ...