[bzoj]2962序列操作
[bzoj]2962序列操作
标签: 线段树
题意
给你一串序列,要你维护三个操作:
1.区间加法
2.区间取相反数
3.区间内任意选k个数相乘的积
题解
第三个操作看起来一脸懵逼啊。
其实是可以合并的。
$ c[o].s[i]=\sum_{j=0}^{20}c[lc].s[j]×c[rc].s[i-j]\(
跟\)C_mn=\sum_{i=0}n C_ni×C_{m-n}{n-i} $这个等式是一个道理的吧。
然后想怎么维护加和取反。
取反比较容易,把取奇数个的答案变成相反数。
加法就稍微复杂一点。
假如p是加上的数,s[i]是区间内取k个数的答案。
\]
Code
(不知道为什么常数很大,bzoj上43s才过)
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<iostream>
#include<algorithm>
#include<cmath>
#include<queue>
#include<stack>
#include<set>
#include<map>
using namespace std;
#define ll long long
#define REP(i,a,b) for(int i=(a),_end_=(b);i<=_end_;i++)
#define DREP(i,a,b) for(int i=(a),_end_=(b);i>=_end_;i--)
#define EREP(i,a) for(int i=start[(a)];i;i=e[i].next)
inline int read()
{
int sum=0,p=1;char ch=getchar();
while(!(('0'<=ch && ch<='9') || ch=='-'))ch=getchar();
if(ch=='-')p=-1,ch=getchar();
while('0'<=ch && ch<='9')sum=sum*10+ch-48,ch=getchar();
return sum*p;
}
const int mod=19940417;
const int maxn=1e5+20;
struct node {
int s[21];
int add,rev;
void Merge(node a,node b)
{
REP(i,0,20)
{
s[i]=0;
REP(j,0,i)s[i]=(s[i]+(ll)a.s[j]*b.s[i-j])%mod;
}
}
};
node c[maxn*4];
int n;
int a[maxn],C[maxn][25];
#define lc (o<<1)
#define rc (o<<1 | 1)
#define left lc,l,mid
#define right rc,mid+1,r
void Reverse(int o,int l,int r)
{
c[o].rev^=1;
c[o].add*=-1;
REP(i,0,20)
if(i & 1)c[o].s[i]*=-1;
}
void Add(int o,int l,int r,ll add)
{
c[o].add=(c[o].add+add)%mod;
DREP(i,min(20,r-l+1),0)
{
ll X=add;
DREP(j,i-1,0)
{
c[o].s[i]=(c[o].s[i]+(ll)c[o].s[j]*X%mod*C[r-l+1-j][i-j])%mod;
X=(ll)X*add%mod;
}
}
}
void pushdown(int o,int l,int r)
{
int mid=(l+r)>>1;
if(c[o].rev)
{
c[o].rev=0;
Reverse(lc,l,mid);
Reverse(rc,mid+1,r);
}
if(c[o].add)
{
Add(lc,l,mid,c[o].add);
Add(rc,mid+1,r,c[o].add);
c[o].add=0;
}
}
void make_tree(int o,int l,int r)
{
if(l==r)
{
c[o].s[0]=1;c[o].s[1]=a[l];
c[o].add=c[o].rev=0;
return;
}
int mid=(l+r)>>1;
make_tree(left);
make_tree(right);
c[o].Merge(c[lc],c[rc]);
}
int q;
void init()
{
n=read();q=read();
REP(i,1,n)a[i]=read();
C[0][0]=1;
REP(i,1,n)
{
C[i][0]=1;
REP(j,1,i)C[i][j]=(C[i-1][j-1]+C[i-1][j])%mod;
}
make_tree(1,1,n);
}
void updatea(int ql,int qr,ll x,int o,int l,int r)
{
if(ql<=l && r<=qr)
{
Add(o,l,r,x);
return;
}
int mid=(l+r)>>1;
pushdown(o,l,r);
if(ql<=mid )updatea(ql,qr,x,left);
if(qr>mid)updatea(ql,qr,x,right);
c[o].Merge(c[lc],c[rc]);
}
void updater(int ql,int qr,int o,int l,int r)
{
if(ql<=l && r<=qr)
{
Reverse(o,l,r);
return;
}
int mid=(l+r)>>1;
pushdown(o,l,r);
if(ql<=mid)updater(ql,qr,left);
if(qr>mid)updater(ql,qr,right);
c[o].Merge(c[lc],c[rc]);
}
node query(int ql,int qr,int o,int l,int r)
{
if(ql<=l && r<=qr)return c[o];
int mid=(l+r)>>1;
pushdown(o,l,r);
if(ql>mid)return query(ql,qr,right);
else if(qr<=mid)return query(ql,qr,left);
else
{
node a=query(ql,mid,left),b=query(mid+1,qr,right);
node c;c.Merge(a,b);
return c;
}
}
void doing()
{
REP(i,1,q)
{
char ch;
//cin>>ch;
scanf("\n%c",&ch);
if(ch=='I')
{
int l=read(),r=read();ll x=read();
updatea(l,r,x,1,1,n);
}
else if(ch=='R')
{
int l=read(),r=read();
updater(l,r,1,1,n);
}else
{
int l=read(),r=read(),k=read();
node x=query(l,r,1,1,n);
printf("%d\n",(x.s[k]+mod)%mod);
}
}
}
int main()
{
freopen("sequence.in","r",stdin);
freopen("sequence.out","w",stdout);
init();
doing();
return 0;
}
[bzoj]2962序列操作的更多相关文章
- bzoj 2962 序列操作
2962: 序列操作 Time Limit: 50 Sec Memory Limit: 256 MB[Submit][Status][Discuss] Description 有一个长度为n的序列, ...
- bzoj 2962 序列操作——线段树(卷积?)
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2962 如果 _,_,_,…… 变成了 (_+k),(_+k),(_+k),…… ,计算就是在 ...
- bzoj 2962 序列操作 —— 线段树
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2962 维护 sum[i] 表示选 i 个的乘积和,合并两个子树就枚举两边选多少,乘起来即可: ...
- bzoj 1858 序列操作
bzoj 1858 序列操作 带有随机多个区间单值覆盖的区间操作题,可考虑用珂朵莉树解决. #include<bits/stdc++.h> using namespace std; #de ...
- 【BZOJ-2962】序列操作 线段树 + 区间卷积
2962: 序列操作 Time Limit: 50 Sec Memory Limit: 256 MBSubmit: 678 Solved: 246[Submit][Status][Discuss] ...
- BZOJ 2962
2962: 序列操作 Time Limit: 50 Sec Memory Limit: 256 MBSubmit: 618 Solved: 225[Submit][Status][Discuss] ...
- bzoj2962 序列操作
2962: 序列操作 Time Limit: 50 Sec Memory Limit: 256 MBSubmit: 1145 Solved: 378[Submit][Status][Discuss ...
- (WAWAWAWAWAWA) BZOJ 1858: [Scoi2010]序列操作
二次联通门 : BZOJ 1858: [Scoi2010]序列操作 /* BZOJ 1858: [Scoi2010]序列操作 已经... 没有什么好怕的的了... 16K的代码... 调个MMP啊.. ...
- bzoj 1858: [Scoi2010]序列操作
1858: [Scoi2010]序列操作 Time Limit: 10 Sec Memory Limit: 64 MB 线段树,对于每个区间需要分别维护左右和中间的1和0连续个数,并在op=4时特殊 ...
随机推荐
- Angular 4 设置组件样式的几种方式
你用Angular吗? 一.介绍 如何只改动最简单的css代码,呈现完全不一样的视图效果. 第一种:最基本的设置: 图1 代码 图2 界面运行效果图 平常,想给一个label或者p等标签添加样 ...
- TCP长连接和短连接的区别
当网络通信时采用TCP协议时,在真正的读写操作之前,server与client之间必须建立一个连接,当读写操作完成后,双方不再需要这个连接时它们可以释放这个连接,连接的建立是需要三次握手的,而释放则需 ...
- vs Code 运行一个本地WEB服务器
VS Code作为前端开发工具,值得推荐,用上之后,爱不释手 当我们建立一个全静态的前端项目时,如果需要将整个项目运行起来也非常简单,两步即可: 1.安装npm install -g live-ser ...
- 1.JavaScript 教程:基础语法
简介: JavaScript web 开发人员必须学习的 3 门语言中的一门: HTML 定义了网页的内容 CSS 描述了网页的布局 JavaScript 网页的行为 用法: (1)HTML 中的脚本 ...
- [Gradle] 在 Eclipse 下利用 gradle 构建系统
转载自:http://www.ibm.com/developerworks/cn/opensource/os-cn-gradle/ 构建系统时候常常要用到 Ant, Maven 等工具,对于初学者 ...
- SQL Server中的变更捕获技术--简单部署
------准备------ CREATE DATABASE db_test_cdc ,) ,name )); INSERT INTO t1(name)VALUES('test') ------开始- ...
- php curl模拟登陆抓取数据
http://www.cnblogs.com/zengguowang/p/6814474.html
- Tomcat xxx unbound
从别的地方import的代码 .出现apache unbound 第一步:选中项目右键Build Path->Configure Build Path-->选中Tomcat 7.0 unb ...
- iphone启动图UI切图尺寸对照保存
- error_reporting
有关error_reporting()函数: error_reporting() 设置 PHP 的报错级别并返回当前级别. ; 错误报告是按位的,或者将数字加起来得到想要的错误报告等级. ; E_AL ...