传送门

终于过了这道题。。

要注意标记之间的影响,和add操作时更新求和的顺序。

same 区间每个数设置为x标记

mult  区间每个数乘x标记

add  区间每个数加x标记

①:当打same标记时,mult标记和add标记就没用了,要初始化。

②:当打mult标记时,add标记也要乘相应的值。

---------------求和可以递推过来---------------

1次方求和就正常求和

即:sum1[rt]=sum1[rt]+(r-l+1)*val;

2次方 可以由 (x+val)= x2+val2+2*x*val

即:sum2[rt]=sum2[rt]+2*sum1[rt]*val+(r-l+1)*val*val;

3次方 可以由 (x+val)= x3+val3+3*x2*val+3*x*val2

即:sum3[rt]=sum3[rt]+(r-l+1)*val*val*val+3*sum2[rt]*val+3*sum1[rt]*val*val;

再次注意  add求和要按照sum3,sum2, sum1的顺序!

算是又对取模有了一点了解。。

a*b*c*d  取模为  (a*b%mod) * (c*d%mod) %mod

a+b*c*d+3*a*b*c*d  取模为 (a + (b*c)%mod*d + 3*((a*b)%mod)*((c*d)%mod)%mod)%mod

 #include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int mod=;
const int maxx=; ll add[maxx<<],mult[maxx<<],same[maxx<<];
ll sum1[maxx<<],sum2[maxx<<],sum3[maxx<<]; int n,m,sum; void pushup(int rt) {
sum1[rt]=(sum1[rt<<]+sum1[rt<<|])%mod;
sum2[rt]=(sum2[rt<<]+sum2[rt<<|])%mod;
sum3[rt]=(sum3[rt<<]+sum3[rt<<|])%mod;
} void build(int l,int r,int rt) {
add[rt]=same[rt]=;
mult[rt]=;
if(l==r) {
sum1[rt]=sum2[rt]=sum3[rt]=;
return;
}
int mid=(l+r)>>;
build(l,mid,rt<<);
build(mid+,r,rt<<|);
pushup(rt);
} void pushdown(int rt,int len) { //下传标记
if(same[rt]) {
sum1[rt<<]=(len-(len>>))*same[rt]%mod;
sum1[rt<<|]=(len>>)*same[rt]%mod;
sum2[rt<<]=((len-(len>>))*same[rt]%mod)*same[rt]%mod;
sum2[rt<<|]=((len>>)*same[rt]%mod)*same[rt]%mod;
sum3[rt<<]=((len-(len>>))*same[rt]%mod)*(same[rt]*same[rt]%mod)%mod;
sum3[rt<<|]=((len>>)*same[rt]%mod)*(same[rt]*same[rt]%mod)%mod;
same[rt<<]=same[rt<<|]=same[rt];
add[rt<<]=add[rt<<|]=; //也需要改变
mult[rt<<]=mult[rt<<|]=; //也需要改变
same[rt]=;
}
if(mult[rt]!=) {
mult[rt<<]=mult[rt<<]*mult[rt]%mod;
mult[rt<<|]=mult[rt<<|]*mult[rt]%mod;
add[rt<<]=add[rt<<]*mult[rt]%mod; //改变
add[rt<<|]=add[rt<<|]*mult[rt]%mod; //改变
sum1[rt<<]=sum1[rt<<]*mult[rt]%mod;
sum1[rt<<|]=sum1[rt<<|]*mult[rt]%mod;
sum2[rt<<]=(sum2[rt<<]*mult[rt]%mod)*mult[rt]%mod;
sum2[rt<<|]=(sum2[rt<<|]*mult[rt]%mod)*mult[rt]%mod;
sum3[rt<<]=(sum3[rt<<]*mult[rt]%mod)*(mult[rt]*mult[rt]%mod)%mod;
sum3[rt<<|]=(sum3[rt<<|]*mult[rt]%mod)*(mult[rt]*mult[rt]%mod)%mod;
mult[rt]=;
}
if(add[rt]) {
add[rt<<]=(add[rt<<]+add[rt])%mod;
add[rt<<|]=(add[rt<<|]+add[rt])%mod;
sum3[rt<<]=(sum3[rt<<]+*sum2[rt<<]*add[rt]%mod+*(sum1[rt<<]*add[rt]%mod)*add[rt]%mod+(add[rt]*add[rt]%mod)*(add[rt]*(len-(len>>))%mod))%mod;
sum3[rt<<|]=(sum3[rt<<|]+*sum2[rt<<|]*add[rt]%mod+*(sum1[rt<<|]*add[rt]%mod)*add[rt]%mod+(add[rt]*add[rt]%mod)*(add[rt]*(len>>)%mod))%mod;
sum2[rt<<]=(sum2[rt<<]+*sum1[rt<<]*add[rt]%mod+(add[rt]*add[rt]%mod)*(len-(len>>))%mod)%mod;
sum2[rt<<|]=(sum2[rt<<|]+*sum1[rt<<|]*add[rt]%mod+(add[rt]*add[rt]%mod)*(len>>)%mod)%mod;
sum1[rt<<]=(sum1[rt<<]+(len-(len>>))*add[rt]%mod)%mod;
sum1[rt<<|]=(sum1[rt<<|]+(len>>)*add[rt]%mod)%mod;
add[rt]=;
}
} void updata(int L,int R, int val,int op,int l,int r,int rt) {
if(L<=l&&R>=r) {
if(op==) {
same[rt]=val;
mult[rt]=; //改变
add[rt]=; //改变
sum1[rt]=(r-l+)*val%mod;
sum2[rt]=((r-l+)*val%mod)*val%mod;
sum3[rt]=((r-l+)*val%mod)*(val*val%mod)%mod;
} else if(op==) {
mult[rt]=mult[rt]*val%mod;
add[rt]=add[rt]*val%mod; //改变
sum1[rt]=sum1[rt]*val%mod;
sum2[rt]=(sum2[rt]*val%mod)*val%mod;
sum3[rt]=(sum3[rt]*val%mod)*(val*val%mod)%mod;
} else if(op==) {
add[rt]=(add[rt]+val)%mod;
sum3[rt]=(sum3[rt]+*sum2[rt]*val%mod+*(sum1[rt]*val%mod)*val%mod+(((r-l+)*val%mod)*(val*val%mod)%mod))%mod;
sum2[rt]=(sum2[rt]+*sum1[rt]*val%mod+((r-l+)*val%mod)*val%mod)%mod;
sum1[rt]=(sum1[rt]+(r-l+)*val%mod)%mod;
}
return;
}
pushdown(rt,r-l+);
int mid=(l+r)>>;
if(L<=mid) updata(L,R,val,op,l,mid,rt<<);
if(R>mid) updata(L,R,val,op,mid+,r,rt<<|);
pushup(rt);
} ll query(int L,int R,int val,int l,int r,int rt) {
if(L<=l&&R>=r) {
if(val==) {
return sum1[rt]%mod;
} else if(val==) {
return sum2[rt]%mod;
} else return sum3[rt]%mod;
}
ll sum=;
pushdown(rt,r-l+);
int mid=(l+r)>>;
if(L<=mid) sum=(sum+query(L,R,val,l,mid,rt<<))%mod;
if(R>mid) sum=(sum+query(L,R,val,mid+,r,rt<<|))%mod;
return sum;
} int main() {
while(~scanf("%d%d",&n,&m)) {
if(n==&&m==) break;
build(,n,);
while(m--) {
int op,x,y,val;
scanf("%d%d%d%d",&op,&x,&y,&val);
if(op==) {
printf("%lld\n",query(x,y,val,,n,));
}
else updata(x,y,val,op,,n,);
}
}
}

HDU4578 Transformation (多操作线段树)的更多相关文章

  1. COGS 2638. 数列操作ψ 线段树

    传送门 : COGS 2638. 数列操作ψ 线段树 这道题让我们维护区间最大值,以及维护区间and,or一个数 我们考虑用线段树进行维护,这时候我们就要用到吉司机线段树啦 QAQ 由于发现若干次an ...

  2. 【题解】P4247 [清华集训]序列操作(线段树修改DP)

    [题解]P4247 [清华集训]序列操作(线段树修改DP) 一道神仙数据结构(DP)题. 题目大意 给定你一个序列,会区间加和区间变相反数,要你支持查询一段区间内任意选择\(c\)个数乘起来的和.对1 ...

  3. 【BZOJ-2962】序列操作 线段树 + 区间卷积

    2962: 序列操作 Time Limit: 50 Sec  Memory Limit: 256 MBSubmit: 678  Solved: 246[Submit][Status][Discuss] ...

  4. 【BZOJ-1858】序列操作 线段树

    1858: [Scoi2010]序列操作 Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 1961  Solved: 991[Submit][Status ...

  5. BZOJ 1858: [Scoi2010]序列操作( 线段树 )

    略恶心的线段树...不过只要弄清楚了AC应该不难.... ---------------------------------------------------------------- #inclu ...

  6. 【bzoj1858】[Scoi2010]序列操作 线段树区间合并

    题目描述 lxhgww最近收到了一个01序列,序列里面包含了n个数,这些数要么是0,要么是1,现在对于这个序列有五种变换操作和询问操作: 0 a b 把[a, b]区间内的所有数全变成0 1 a b ...

  7. 【BZOJ2962】序列操作 线段树

    [BZOJ2962]序列操作 Description 有一个长度为n的序列,有三个操作1.I a b c表示将[a,b]这一段区间的元素集体增加c,2.R a b表示将[a,b]区间内所有元素变成相反 ...

  8. Luogu P2572 [SCOI2010]序列操作 线段树。。

    咕咕了...于是借鉴了小粉兔的做法ORZ... 其实就是维护最大子段和的线段树,但上面又多了一些操作....QWQ 维护8个信息:1/0的个数(sum),左/右边起1/0的最长长度(ls,rs),整段 ...

  9. BZOJ 4034 [HAOI2015]树上操作 线段树+树剖或dfs

    题意 直接照搬原题面 有一棵点数为 N 的树,以点 1 为根,且树点有边权.然后有 M 个 操作,分为三种: 操作 1 :把某个节点 x 的点权增加 a . 操作 2 :把某个节点 x 为根的子树中所 ...

随机推荐

  1. 【转载】linux进程及进程控制

    Linux进程控制   程序是一组可执行的静态指令集,而进程(process)是一个执行中的程序实例.利用分时技术,在Linux操作系统上同时可以运行多个进程.分时技术的基本原理是把CPU的运行时间划 ...

  2. 【转载】TCP演进简述

    TCP演进简述 http://www.cnblogs.com/fll/ 一.互联网概述 TCP,即传输控制协议,是目前网络上使用的最多的传输协议,我们知道,整个互联网的体系结构是以IP协议提供的无连接 ...

  3. JavaSE_12_Properties类和缓冲流

    1.Properties类 java.util.Properties 继承于Hashtable ,来表示一个持久的属性集.它使用键值结构存储数据,每个键及其对应值都是一个字符串.该类也被许多Java类 ...

  4. centos7 搭建 php7 + nginx (2)

    安装php php下载地址 # 避免出错,先安装下面 yum install libzip libzip-devel libxml2-devel openssl openssl-devel bzip2 ...

  5. 深入浅出 Java Concurrency (33): 线程池 part 6 线程池的实现及原理 (1)[转]

    线程池数据结构与线程构造方法 由于已经看到了ThreadPoolExecutor的源码,因此很容易就看到了ThreadPoolExecutor线程池的数据结构.图1描述了这种数据结构. 图1 Thre ...

  6. PAT甲级——A1078 Hashing

    The task of this problem is simple: insert a sequence of distinct positive integers into a hash tabl ...

  7. Neo4j删除节点和关系、彻底删除节点标签名

    https://www.jianshu.com/p/59bd829de0de 总结提前: [1]先删关系,再删节点 [2]当记不得关系名时,type(r)可以查到关系名 [3]彻底删除节点标签名,需要 ...

  8. WPF界面设计中常用的一些代码片段及属性

    一.窗体去掉标题栏.消除掉标题栏后的白边,把窗体置于屏幕中间,窗口大小不能改变. WindowStyle="None" AllowsTransparency="True& ...

  9. 搞了一宿,弄完了一个POP3协议

    POP3协议和SMTP协议都会了,加上PE文件的读写,APIHOOK,以及远程进程注入,我是不是就可以写个简单点的通过邮件传播的蠕虫病毒了,哈哈哈哈哈哈. 感觉POP3协议挺简单的,比那个该死的SMT ...

  10. 2018-12-18-WPF-一个空的-WPF-程序有多少个窗口

    title author date CreateTime categories WPF 一个空的 WPF 程序有多少个窗口 lindexi 2018-12-18 21:16:40 +0800 2018 ...