bzoj2962 序列操作
2962: 序列操作
Time Limit: 50 Sec Memory Limit: 256 MB
Submit: 1145 Solved: 378
[Submit][Status][Discuss]
Description
有一个长度为n的序列,有三个操作1.I a b c表示将[a,b]这一段区间的元素集体增加c,2.R a b表示将[a,b]区间内所有元素变成相反数,3.Q a b c表示询问[a,b]这一段区间中选择c个数相乘的所有方案的和mod 19940417的值。
Input
第一行两个数n,q表示序列长度和操作个数。
第二行n个非负整数,表示序列。
接下来q行每行输入一个操作I a b c或者 R a b或者Q a b c意义如题目描述。
Output
对于每个询问,输出选出c个数相乘的所有方案的和mod19940417的值。
Sample Input
1 2 3 4 5
I 2 3 1
Q 2 4 2
R 1 5
I 1 3 -1
Q 1 5 1
Sample Output
19940397
样例说明
做完第一个操作序列变为1 3 4 4 5。
第一次询问结果为3*4+3*4+4*4=40。
做完R操作变成-1 -3 -4 -4 -5。
做完I操作变为-2 -4 -5 -4 -5。
第二次询问结果为-2-4-5-4-5=-20。
HINT
100%的数据n<=50000,q<=50000,初始序列的元素的绝对值<=109,I a b c中保证[a,b]是一个合法区间,|c|<=109,R a b保证[a,b]是个合法的区间。Q a b c中保证[a,b]是个合法的区间1<=c<=min(b-a+1,20)。
Source
,感性理解一下就是c的k次方,c要占k个位置,而另外的位置已经被f选的数给占了,只能在剩下的位置中选c,这大概就是组合数的意义.剩下的都可以通过合并同类项得到.#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm> using namespace std; typedef long long ll;
const ll maxn = ,mod = ;
ll n,q;
ll a[maxn],c[maxn][]; struct node
{
ll add,cover,f[],L,R;
void init()
{
add = cover = L = R = ;
memset(f,,sizeof(f));
f[] = ;
}
} e[maxn << ]; node pushup(node a,node b)
{
node c;
c.init();
c.L = a.L;
c.R = b.R;
ll len1 = c.R - c.L + ;
ll len2 = a.R - a.L + ;
ll len3 = b.R - b.L + ;
for (ll i = ; i <= min(len2,1LL * ); i++)
for (ll j = ; j <= min(len3,1LL * ); j++)
{
if (i + j > )
break;
c.f[i + j] = (c.f[i + j] + a.f[i] * b.f[j] % mod) % mod;
}
c.f[] = ; //易错点
return c;
} void fan(ll o)
{
ll len = e[o].R - e[o].L + ;
for (ll i = ; i <= min(len,1LL * ); i++)
{
if (i % == )
{
e[o].f[i] = -e[o].f[i];
e[o].f[i] = (e[o].f[i] + mod) % mod;
}
}
e[o].cover ^= ;
e[o].add = -e[o].add;
e[o].add = (e[o].add + mod) % mod; //取反后一定要变成正数
} void jia(ll o,ll v)
{
ll len = e[o].R - e[o].L + ;
for (ll i = min(len,1LL * );i >= ; i--) //一定要倒着推
{
ll k = v;
for (ll j = i - ; j >= ; j--)
{
e[o].f[i] = (e[o].f[i] + e[o].f[j] * c[len - j][i - j] % mod * k % mod) % mod;
k = k * v % mod;
}
}
e[o].add = (e[o].add + v) % mod;
} void pushdown(ll o)
{
if (e[o].cover)
{
fan(o * );
fan(o * + );
e[o].cover = ;
}
if (e[o].add)
{
jia(o * ,e[o].add);
jia(o * + ,e[o].add);
e[o].add = ;
}
} void build(ll o,ll l,ll r)
{
e[o].init();
e[o].L = l,e[o].R = r;
if (l == r)
{
e[o].f[] = a[l] % mod;
return;
}
ll mid = (l + r) >> ;
build(o * ,l,mid);
build(o * + ,mid + ,r);
e[o] = pushup(e[o * ],e[o * + ]);
} void update1(ll o,ll l,ll r,ll x,ll y,ll v)
{
if (x <= l && r <= y)
{
jia(o,v);
return;
}
pushdown(o);
ll mid = (l + r) >> ;
if (x <= mid)
update1(o * ,l,mid,x,y,v);
if (y > mid)
update1(o * + ,mid + ,r,x,y,v);
e[o] = pushup(e[o * ],e[o * + ]);
} void update2(ll o,ll l,ll r,ll x,ll y)
{
if (x <= l && r <= y)
{
fan(o);
return;
}
pushdown(o);
ll mid = (l + r) >> ;
if (x <= mid)
update2(o * ,l,mid,x,y);
if (y > mid)
update2(o * + ,mid + ,r,x,y);
e[o] = pushup(e[o * ],e[o * + ]);
} node query(ll o,ll l,ll r,ll x,ll y)
{
if (x <= l && r <= y)
return e[o];
pushdown(o);
ll mid = (l + r) >> ;
if (y <= mid)
return query(o * ,l,mid,x,y);
else if (x > mid)
return query(o * + ,mid + ,r,x,y);
else
return pushup(query(o * ,l,mid,x,mid),query(o * + ,mid + ,r,mid + ,y));
} int main()
{
scanf("%lld%lld",&n,&q);
c[][] = ;
for (ll i = ; i <= n; i++)
{
c[i][] = ;
for (ll j = ; j <= ; j++)
c[i][j] = (c[i - ][j] + c[i - ][j - ]) % mod;
}
for (ll i = ; i <= n; i++)
scanf("%lld",&a[i]);
build(,,n);
while (q--)
{
char ch[];
ll a,b,c;
scanf("%s",ch);
if (ch[] == 'I')
{
scanf("%lld%lld%lld",&a,&b,&c);
update1(,,n,a,b,c);
}
if (ch[] == 'R')
{
scanf("%lld%lld",&a,&b);
update2(,,n,a,b);
}
if (ch[] == 'Q')
{
scanf("%lld%lld%lld",&a,&b,&c);
node temp = query(,,n,a,b);
printf("%lld\n",temp.f[c] % mod);
}
} return ;
}
bzoj2962 序列操作的更多相关文章
- [bzoj2962]序列操作_线段树_区间卷积
序列操作 bzoj-2962 题目大意:给定一个n个数的正整数序列,m次操作.支持:1.区间加:2.区间取相反数:3.区间求选c个数的乘积和. 注释:$1\le n,m\le 5\cdot 10^4$ ...
- bzoj2962 序列操作 题解
题目大意: 有一个长度为n的序列,有三个操作1.I a b c表示将[a,b]这一段区间的元素集体增加c,2.R a b表示将[a,b]区间内所有元素变成相反数,3.Q a b c表示询问[a,b]这 ...
- 2019.01.04 bzoj2962: 序列操作(线段树+组合数学)
传送门 线段树基础题. 题意:要求维护区间区间中选择ccc个数相乘的所有方案的和(c≤20c\le20c≤20),支持区间加,区间取负. 由于c≤20c\le20c≤20,因此可以对于每个线段树节点可 ...
- 【BZOJ2962】序列操作(线段树)
[BZOJ2962]序列操作(线段树) 题面 BZOJ 题解 设\(s[i]\)表示区间内选择\(i\)个数的乘积的和 考虑如何向上合并? \(s[k]=\sum_{i=0}^klson.s[i]*r ...
- 【BZOJ2962】序列操作 线段树
[BZOJ2962]序列操作 Description 有一个长度为n的序列,有三个操作1.I a b c表示将[a,b]这一段区间的元素集体增加c,2.R a b表示将[a,b]区间内所有元素变成相反 ...
- 【BZOJ-2962】序列操作 线段树 + 区间卷积
2962: 序列操作 Time Limit: 50 Sec Memory Limit: 256 MBSubmit: 678 Solved: 246[Submit][Status][Discuss] ...
- Python通用序列操作
1.序列概览 1.数据结构 序列.容器 Python中最基本的数据结构是序列,其有索引(从左到右第一个索引为0,从右到左第一个索引为-1). Python包含6中内建的序列: 列表 元组 字符串 Un ...
- 【BZOJ-1858】序列操作 线段树
1858: [Scoi2010]序列操作 Time Limit: 10 Sec Memory Limit: 64 MBSubmit: 1961 Solved: 991[Submit][Status ...
- bzoj 1858: [Scoi2010]序列操作
1858: [Scoi2010]序列操作 Time Limit: 10 Sec Memory Limit: 64 MB 线段树,对于每个区间需要分别维护左右和中间的1和0连续个数,并在op=4时特殊 ...
随机推荐
- Spring Cloud(三):服务提供与调用 Eureka【Finchley 版】
Spring Cloud(三):服务提供与调用 Eureka[Finchley 版] 发表于 2018-04-15 | 更新于 2018-05-07 | 上一篇文章我们介绍了 Eureka 服务 ...
- JavaScript里的循环方法之forEach,for-in,for-of
JavaScript一种直译式脚本语言,是一种动态类型.弱类型.基于原型的语言,内置支持类型.它的解释器被称为JavaScript引擎,为浏览器的一部分,广泛用于客户端的脚本语言,最早是在HTML(标 ...
- 【springmvc+mybatis项目实战】杰信商贸-6.重点知识回顾
1.重点知识回顾 Maven1)覆盖仓库文件,实际企业开发,公司会架一个测试服务器,在测试服务器中架私服.我们开发人员的程序,都连接私服.当本地没有项目中要使用的jar,Myeclipse maven ...
- Spring学习(1):侵入式与非侵入式,轻量级与重量级
一. 引言 在阅读spring相关资料,都会提到Spring是非侵入式编程模型,轻量级框架,那么就有必要了解下这些概念. 二. 侵入式与非侵入式 非侵入式:使用一个新的技术不会或者基本不改变原有代码结 ...
- Teaching Machines to Understand Us 让机器理解我们 之一 引言
Teaching Machines to Understand Us By Tom Simonite MIT Technology Review Vol.118 No.5 2015 让机器理解我 ...
- Halcon如何保存仿射变换矩阵
这里我们通过序列化来实现的,如下图,写到硬盘的HomMat2D_1内容和从硬盘里HomMat2D_2读出的内容一致,源代码在图片下方. Halcon源代码: hom_mat2d_identity (H ...
- 使用Scrapy构建一个网络爬虫
记得n年前项目需要一个灵活的爬虫工具,就组织了一个小团队用Java实现了一个爬虫框架,可以根据目标网站的结构.地址和需要的内容,做简单的配置开发,即可实现特定网站的爬虫功能.因为要考虑到各种特殊情形, ...
- hadoop 中balance 机制
Hadoop的HDFS集群非常容易出现机器与机器之间磁盘利用率不平衡的情况,比如集群中添加新的数据节点.当HDFS出现不平衡状况的时候,将引发很多问题,比如MR程序无法很好地利用本地计算的优势,机器之 ...
- USACO 2.4.4 Bessie Come Home 回家(最短路)
Description 现在是晚餐时间,而母牛们在外面分散的牧场中. 农民约翰按响了电铃,所以她们开始向谷仓走去. 你的工作是要指出哪只母牛会最先到达谷仓(在给出的测试数据中,总会有且只有一只速度最快 ...
- 如何利用Xshell在Linux下安装jdk
本文会详细介绍如何在Linux下安装JDK1.8 首先要设置虚拟机的IP地址,不知道如何设置的话可以 翻看我的前一篇博客 http://www.cnblogs.com/xiaoxiaoSMILE/ ...