An easy problem C

Time Limit: 4000/2000MS (Java/Others) Memory Limit: 65535/65535KB (Java/Others)

Problem Description

N个数排成一列,有三种操作。1.给一段区间内的每个数乘上一个非负整数。2.给一段区间内的每个数加上一个非负整数.3.询问一段区间的和模上P的值。

Input

第一行两个整数N(1≤N≤100000)表示数的个数,P(1≤P≤1000000000)表示模的值。接下来一行N个整数ai(0≤ai≤1000000000),接下来一行一个整数M(1≤M≤100000)表示操作数量,接下来M行每行描述一个操作。第一种操作描述:1 L R C(0≤C≤1000000000),表示把L到R这段区间每个数乘上一个C。第二种操作描述:2 L R C(0≤C≤1000000000),表示把L到R这段区间每个数加上一个C。第三种操作3 L R 表示询问L到R这段区间内的数的和模上P的值。

Output

对面每个询问,输出对应的答案,每个询问占一行。

Sample Input

7 43

1 2 3 4 5 6 7

5

1 2 5 5

3 2 4

2 3 7 9

3 1 3

3 4 7

Sample Output

2

35

8


解题心得:

  1. 这是一个区间更新的线段树,写起来很烦。写这个线段树先说几点。
  2. 首先是lazy标记。这个lazy标记标记的是一个区间,它将整个区间标记,但是它并不对这个区间的子树进行处理,只有在要使用这个区间的子树的时候再将子树更新,lazy标记下移,这样在处理区间更新的时候时间复杂度就是O(logn)。lazy标记的特点就是懒惰,它只是将当前标记,并不下移,当要使用下面的子树的时候才下移。这里还要注意的一点是,lazy标记下移之后在这个区间的lazy标记应当取消,这点初学者很容易忘记。
  3. 然后就是这个题的处理了,比较麻烦,它要求将一个区间加上一个数,或者乘上一个数,每次加或者乘的时候都要将每个区间的sum给更新一下,还要向上维护。当前的sum要马上更新,而这个时候的乘上的总积以及加上的总和只是看作一个lazy标记,在要用到的时候下移。还有就是乘和加的优先级问题,应该先乘上在加。
  4. 这个题对于向上维护和向下维护的要求比较高,不注意会产生很多的bug。

//这个题为了防止溢出还是全部写long long吧

#include<bits/stdc++.h>
using namespace std;
const int maxn = 1e5+100;
typedef long long ll;
ll p,ans;
struct node
{
ll l,r,sum,mul,plus;
} tree[maxn*4]; //向上更新
void pushup(ll root)
{
tree[root].sum = (tree[root<<1].sum + tree[root<<1|1].sum)%p;
} //来自标记下移
void pushdown(ll root,ll mid,ll l,ll r)
{
if(tree[root].mul == 1 && tree[root].plus == 0)//当这个区间没有被标记的时候就不用下移
return ;
tree[root<<1].mul = (tree[root<<1].mul * tree[root].mul)%p;
tree[root<<1].plus = ((tree[root<<1].plus * tree[root].mul)%p + tree[root].plus)%p;
tree[root<<1].sum = ((tree[root<<1].sum * tree[root].mul)%p + tree[root].plus *(mid - l + 1)%p)%p; tree[root<<1|1].mul = (tree[root<<1|1].mul * tree[root].mul)%p;
tree[root<<1|1].plus = ((tree[root<<1|1].plus * tree[root].mul)%p + tree[root].plus)%p;
tree[root<<1|1].sum = ((tree[root<<1|1].sum * tree[root].mul)%p + tree[root].plus *(r - mid)%p)%p; //要将当前的lazy标记给取消
tree[root].mul = 1;
tree[root].plus = 0;
} //先建一个树
void build_tree(ll l,ll r,ll root)
{
tree[root].l = l,tree[root].r = r;
if(l == r)
{
scanf("%lld",&tree[root].sum);
tree[root].sum = tree[root].sum % p;
tree[root].plus = 0;
tree[root].mul = 1;
return ;
}
ll mid = (l + r) / 2;
pushdown(root,mid,l,r);
build_tree(l,mid,root<<1);
build_tree(mid+1,r,root<<1|1);
tree[root].plus = 0,tree[root].mul = 1;
pushup(root);
} //区间乘一个数
void querymu(ll L,ll R,ll l,ll r,ll mu,ll root)
{
if(L <= l && R >= r)
{
tree[root].mul = (tree[root].mul * mu)%p;//先乘再加
tree[root].plus = (tree[root].plus * mu)%p;
tree[root].sum = (tree[root].sum * mu)%p;
return ;
}
ll mid = (l + r) >> 1;
pushdown(root,mid,l,r);//lazy标记下移
if(R <= mid)
querymu(L,R,l,mid,mu,root<<1);
else if(L > mid)
querymu(L,R,mid+1,r,mu,root<<1|1);
else
{
querymu(L,mid,l,mid,mu,root<<1);
querymu(mid+1,R,mid+1,r,mu,root<<1|1);
}
pushup(root);//向上更新
} //区间加一个数
void queryPlus(ll L,ll R,ll l,ll r,ll Plus,ll root)
{
if(L <= l && R >= r)
{
tree[root].plus = (tree[root].plus + Plus)%p;
tree[root].sum = (tree[root].sum + ((R - L + 1) * Plus)%p)%p;//这里要注意一下,每个数加上一个数,总共有R-L+1个数
return ;
}
ll mid = (l + r)>>1;
pushdown(root,mid,l,r);//lazy标记下移
if(R <= mid)
queryPlus(L,R,l,mid,Plus,root<<1);
else if(L > mid)
queryPlus(L,R,mid+1,r,Plus,root<<1|1);
else
{
queryPlus(L,mid,l,mid,Plus,root<<1);
queryPlus(mid+1,R,mid+1,r,Plus,root<<1|1);
}
pushup(root);//向上更新
} //查询当前区间的值
long long query(ll L,ll R,ll l,ll r,ll root)
{
if(L <= l && R >= r)
return tree[root].sum % p;
ll mid = (l + r) >> 1;
pushdown(root,mid,l,r);//反正在每一次向下找的时候都要注意lazy标记下移
if(R <= mid)
return query(L,R,l,mid,root<<1)%p;
else if(L > mid)
return query(L,R,mid+1,r,root<<1|1)%p;
else
return ((query(L,mid,l,mid,root<<1)%p) + (query(mid+1,R,mid+1,r,root<<1|1)%p))%p;
} int main()
{
ll n,m;
while(scanf("%lld%lld",&n,&p) != EOF)
{
build_tree(1,n,1);
scanf("%lld",&m);
while(m--)
{
ll a;
scanf("%lld",&a);
if(a == 1)
{
ll L,R,mu;
scanf("%lld%lld%lld",&L,&R,&mu);
querymu(L,R,1,n,mu%p,1);
}
if(a == 2)
{
ll L,R,Plus;
scanf("%lld%lld%lld",&L,&R,&Plus);
queryPlus(L,R,1,n,Plus%p,1);
}
if(a == 3)
{
ll L,R;
scanf("%lld%lld",&L,&R);
ans = query(L,R,1,n,1)%p;
printf("%lld\n",ans);
}
}
}
return 0;
}

线段树:CDOJ1597-An easy problem C(区间更新的线段树)的更多相关文章

  1. 线段树:CDOJ1592-An easy problem B (线段树的区间合并)

    An easy problem B Time Limit: 2000/1000MS (Java/Others) Memory Limit: 65535/65535KB (Java/Others) Pr ...

  2. 线段树:CDOJ1591-An easy problem A (RMQ算法和最简单的线段树模板)

    An easy problem A Time Limit: 1000/1000MS (Java/Others) Memory Limit: 65535/65535KB (Java/Others) Pr ...

  3. K - Transformation-hdu 4578(多操作混合区间更新)线段树

    题意:有四种操作 1,  区间 [l, r] 的值都加上 C 2,  区间 [l, r] 的值都乘上 C 3,  区间 [l, r] 的值都变为C 4,  求区间 [l, r]所有数的p次方的和 分析 ...

  4. hdu4348区间更新的主席树+标记永久化

    http://acm.hdu.edu.cn/showproblem.php?pid=4348 sb的标记永久化即可,刚开始add和sum没复制过来wa了两发...,操作和原来的都一样,出来单点变成区间 ...

  5. POJ-3468-A Simple Problem with Integers(区间更新,求和)-splay或线段树

    区间更新求和 主要用来练习splay树区间更新问题 //splay树的题解 // File Name: 3468-splay.cpp // Author: Zlbing // Created Time ...

  6. hdu3966 树链剖分点权模板+线段树区间更新/树状数组区间更新单点查询

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

  7. hdu 5475 An easy problem(暴力 || 线段树区间单点更新)

    http://acm.hdu.edu.cn/showproblem.php?pid=5475 An easy problem Time Limit: 8000/5000 MS (Java/Others ...

  8. UESTC 1591 An easy problem A【线段树点更新裸题】

    An easy problem A Time Limit: 2000/1000MS (Java/Others)     Memory Limit: 65535/65535KB (Java/Others ...

  9. POJ 3468 A Simple Problem with Integers(线段树区间更新区间查询)

    A Simple Problem with Integers Time Limit: 5000MS   Memory Limit: 131072K Total Submissions: 92632   ...

随机推荐

  1. 关系型数据库---MySQL---数据库设计三大范式

    1.第一范式: 1.1.1 数据表的每个数据列具有原子性: 1.1.2 同一个数据表中内容相似的数据列必须消除: 2.第二范式: 第一范式的基础上,每个数据表只描述一件事: 3.第三范式: 第二范式的 ...

  2. CentOS7.2安装MySql5.7并开启远程连接授权

    1.安装mysql5.7 CentOS 7之后的版本yum的默认源中使用MariaDB替代原先MySQL,因此安装方式较为以往有一些改变: 下载mysql的源 wget http://dev.mysq ...

  3. Web可用性设计的247条指导方针

    首页可用性设计 首页元素要清晰的关注用户的关键任务(避免“增加功能倾向(featuritis)”) 如果网站比较大,那么首页应包含搜索输入框 首页要十分清楚的提供产品(内容)分类 在首页或首页内一次点 ...

  4. c#基础 base和this的区别,在继承上面

    base public Person(string name, int age, char gender) { this.Name = name; this.Age = age; this.Gende ...

  5. (wp8.1)样式资源

    在开发wp的时候,难免要设置好多属性.但是有好多属性是重复的,我们可不可统一 设置样式呢,答案是肯定的 代码如下 <Page x:Class="Blue.MainPage" ...

  6. Spring事务的5种隔离级别

    概述:isolation设定事务的隔离级别,事务管理器根据它来控制另外一个事务可以看到本事务内的哪些数据. 定义的5个不同的事务隔离级别: DEFAULT:默认的隔离级别,使用数据库默认的事务隔离级别 ...

  7. AngularJS(二):ng-app指令、表达式

    本文也同步发表在我的公众号“我的天空” ng-app指令 AngularJS指令是扩展的HTML属性,所有指令均带有前缀“ng-”,我们学习的第一个指令便是ng-app,其定义了AngularJS管理 ...

  8. iOS开发 - Protocol协议及委托代理(Delegate)

    因为Object-C是不支持多继承的,所以很多时候都是用Protocol(协议)来代替.Protocol(协议)只能定义公用的一套接口,但不能提供具体的实现方法.也就是说,它只告诉你要做什么,但具体怎 ...

  9. db2数据库创建索引,删除索引,查看表索引,SQL语句执行计划以及优化建议

    1.建立表索引 create index 索引名 on 表名(列名,列名); 2.删除表索引 drop index 索引名 on 表名; 3.查看表索引 select * from sysibm.sy ...

  10. python+selenium之断言Assertion

    一.断言方法 断言是对自动化测试异常情况的判断. # -*- coding: utf-8 -*- from selenium import webdriver import unittest impo ...