题目大意:

题意:有一个序列,有四种操作:

1:区间[l,r]内的数全部加c。

2:区间[l,r]内的数全部乘c。

3:区间[l,r]内的数全部初始为c。

4:询问区间[l,r]内所有数的P次方之和。

这里p可以等于1,2,3三种情况,所以我们需要建立3个数组,当然这里其实只用一个sum[4*N][3]的2维数组其实更好

to[],add[],mul[]为三个懒惰标记to[]先于另外两个,每次做完to[],另外两个标记就要将其初始化

对于mul[]来说,每次执行,add[cur]*=mul[cur]也要随之增加

因为这道题目数过大,需要求mod 10007的值

因为这个我把代码从上午看到了下午简直要疯了

int最大值2147483647

你必须每执行一步操作都需要mod一次,*2后,10007*2*10007 int还不至于爆掉,但也差不多

但是若*3之后那肯定要是再碰到一个超大数,那就GG了

所以每次乘法算完都mod一次吧,每次做mod带上括号是一个好习惯~~

 //这鬼题目里面最好每执行一次乘法就mod一次,不然可能int要爆掉导致报错,尤其是我在fun_add()中改错改了一天才发现是这个原因
#include <cstdio>
#include <cstring>
using namespace std;
#define N 100005
#define mod 10007
int sum[*N][],add[*N],to[*N],mul[*N];
void update(int cur)
{
sum[cur][]=(sum[cur<<][]+sum[cur<<|][])%mod;
sum[cur][]=(sum[cur<<][]+sum[cur<<|][])%mod;
sum[cur][]=(sum[cur<<][]+sum[cur<<|][])%mod;
}
void build(int cur,int x,int y)
{
int mid=(x+y)/,ls=cur<<,rs=cur<<|;
add[cur]=,to[cur]=,mul[cur]=;
if(x==y){
sum[cur][]=sum[cur][]=sum[cur][]=;
return;
}
build(ls,x,mid);
build(rs,mid+,y);
update(cur);
}
void fun_mul(int cur,int x,int y,int val)
{
mul[cur]*=val;
mul[cur]%=mod;
add[cur]*=val;
add[cur]%=mod; int t1=val*val%mod;
int t2=t1*val%mod;
sum[cur][]=sum[cur][]*val;
sum[cur][]%=mod; sum[cur][]=sum[cur][]*t1;
sum[cur][]%=mod; sum[cur][]=sum[cur][]*t2;
sum[cur][]%=mod;
}
void fun_add(int cur,int x,int y,int val)
{
add[cur]+=val;
add[cur]%=mod;
int t1=sum[cur][];
int t2=sum[cur][];
int tmp1=val*val%mod;
int tmp2=val*tmp1%mod;
sum[cur][]+=(y-x+)*val;
sum[cur][]%=mod; sum[cur][]+=(y-x+)*tmp1%mod+((*val)%mod)*t1%mod;
sum[cur][]%=mod; sum[cur][]+=((*val)%mod)*t2%mod+*tmp1*t1%mod+(y-x+)*tmp2%mod;
sum[cur][]%=mod;
}
void fun_to(int cur,int x,int y,int val)
{
mul[cur]=,add[cur]=,to[cur]=val%mod;
int t1=val*val%mod;
int t2=t1*val%mod;
sum[cur][]=(y-x+)*val%mod;
sum[cur][]=(y-x+)*t1%mod;
sum[cur][]=(y-x+)*t2%mod;
}
void pushdown(int cur,int x,int y)
{
int mid=(x+y)/,ls=cur<<,rs=cur<<|;
if(to[cur]){
//to[ls]=to[rs]=to[cur];
fun_to(ls,x,mid,to[cur]);
fun_to(rs,mid+,y,to[cur]);
to[cur]=;
}
if(mul[cur]>){
//mul[ls]*=mul[cur],mul[rs]*=mul[cur];
fun_mul(ls,x,mid,mul[cur]);
fun_mul(rs,mid+,y,mul[cur]);
mul[cur]=;
}
if(add[cur]){
//add[ls]+=add[cur],add[rs]+=add[cur];
fun_add(ls,x,mid,add[cur]);
fun_add(rs,mid+,y,add[cur]);
add[cur]=;
}
}
void change(int cur,int x,int y,int s,int t,int op,int v)
{
int mid=(x+y)/,ls=cur<<,rs=cur<<|;
if(x>=s&&y<=t){
if(op==) fun_add(cur,x,y,v);
if(op==) fun_mul(cur,x,y,v);
if(op==) fun_to(cur,x,y,v);
return;
}
pushdown(cur,x,y);
if(mid>=s) change(ls,x,mid,s,t,op,v);
if(mid+<=t) change(rs,mid+,y,s,t,op,v);
update(cur);
}
void query(int cur,int x,int y,int s ,int t,int p,int &ans)
{
int mid=(x+y)/,ls=cur<<,rs=cur<<|;
if(x>=s&&y<=t){
ans+=sum[cur][p-];
ans%=mod;
return;
}
pushdown(cur,x,y);
if(mid>=s) query(ls,x,mid,s,t,p,ans);
if(mid+<=t) query(rs,mid+,y,s,t,p,ans);
}
int main()
{
int n,m,op,x,y,p;
while(scanf("%d%d",&n,&m),n+m){
build(,,n);
for(int i=;i<m;i++){
scanf("%d%d%d%d",&op,&x,&y,&p);
if(op==){
int ans=;
//printf("%d\n",0);
query(,,n,x,y,p,ans);
printf("%d\n",ans%mod);
}
else change(,,n,x,y,op,p);
}
}
}

HDU 4578 线段树复杂题的更多相关文章

  1. K - Transformation HDU - 4578 线段树经典题(好题)

    题意:区间  加   变成定值 乘  区间查询:和 平方和 立方和 思路:超级超级超级麻烦的一道题  设3个Lazy 标记分别为  change 改变mul乘 add加  优先度change>m ...

  2. hdu 4578 线段树(标记处理)

    Transformation Time Limit: 15000/8000 MS (Java/Others)    Memory Limit: 65535/65536 K (Java/Others) ...

  3. HDU 4578 线段树玄学算法?

    Transformation 题目链接 http://acm.hdu.edu.cn/showproblem.php?pid=4578 Problem Description Yuanfang is p ...

  4. HDU 4893 线段树裸题

    Wow! Such Sequence! Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Oth ...

  5. hdu 1754 线段树模板题

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1754 #include <cstdio> #include <cmath> # ...

  6. HDU - 4578 线段树+三重操作

    这道题自己写了很久,还是没写出来,也看了很多题解,感觉多数还是看的迷迷糊糊,最后面看到一篇大佬的才感觉恍然大悟. 先上一篇大佬的题解:https://blog.csdn.net/aqa20372995 ...

  7. hdu 4578 线段树 ****

    链接:点我  1

  8. hdu 1754 线段树 水题 单点更新 区间查询

    I Hate It Time Limit: 9000/3000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total S ...

  9. hdu 1754 I Hate It 线段树基础题

    Problem Description 很多学校流行一种比较的习惯.老师们很喜欢询问,从某某到某某当中,分数最高的是多少. 这让很多学生很反感. 不管你喜不喜欢,现在需要你做的是,就是按照老师的要求, ...

随机推荐

  1. Java_面向对象中的this和super用法

    this: 1.使用在类中,可以用来修饰属性.方法.构造器 2.表示当前对象或者是当前正在创建的对象 3.当形参与成员变量重名时,如果在方法内部需要使用成员变量,必须添加 this 来表明该变量时类成 ...

  2. 【转】Iconfont

    Iconfont Iconfont 是指用字体文件取代图片文件,来展示图标.特殊字体等元素的一种方法.很多网站都会用它,比如手淘.新浪微博等. 在使用它之前,先来了解一下它的优缺点: 优点:(1)文件 ...

  3. python中的格式化字符

    python中的格式化字符在python中我们会遇到一个问题,问题是如何输出格式化的字符串.我们经常会输出类似'亲爱的xxx你好!你xx月的话费是xx,余额是xx'之类的字符串,而xxx的内容都是根据 ...

  4. Proteus与Keil连接及其仿真(有例子哦!)

    记录一下Proteus仿真的一些设置和使用,以方便自己以后复习和大家交流!如有错误,希望大家指正. 1.Proteus软件的安装,这里就不作说明了.

  5. Sql Server数据库对象访问权限控制

    以下内容主要针对database层面的数据访问权限(比如select, insert, update, delete, execute…) 1.直接给user权限GRANT EXECUTE TO [u ...

  6. MFC技术积累——基于MFC对话框类的那些事儿2

    3. 绘图 3.1 对话框资源编辑 首先通过添加控件的方式来创建一个简单的绘图对话框如图所示,创建步骤为: 第一.在VC++6.0软件环境的灰色空白区域右击,选中Controls,然后会弹出一个控件对 ...

  7. 11G RAC环境数据库启动和关闭

    一步启动Oracle (1) 启动整个集群 # ./crsctl start cluster -all -all选项启动整个集群. 不加-all选项只能启动本节点的服务. (2) 启动本节点集群 以下 ...

  8. Android(java)学习笔记157:开源框架的文件上传(只能使用Post)

    1.文件上传给服务器,服务器端必然要写代码进行支持,如下: 我们新建一个FileUpload.jsp的动态网页,同时我们上传文件只能使用post方式(不可能将上传数据拼凑在url路径下),上传数据Ap ...

  9. linux_2

    mac和linux上默认安装着SSH客户端,Windows上需要自己安装个软件. Windows下SSH客户端的安装:建议从官方网站下载正式程序安装(免费) Putty:https://www.chi ...

  10. mybatis-spring_缓存

    学习之前需要先了解一下什么是mybatis一级缓存? LZ推荐:https://blog.csdn.net/niunai112/article/details/80601793#%E4%B8%80%E ...