LOJ #6283. 数列分块入门 7-分块(区间乘法、区间加法、单点查询)
题目描述
给出一个长为 nn 的数列,以及 nn 个操作,操作涉及区间乘法,区间加法,单点询问。
输入格式
第一行输入一个数字 nn。
第二行输入 nn 个数字,第 ii 个数字为 a_iai,以空格隔开。
接下来输入 nn 行询问,每行输入四个数字 \mathrm{opt}opt、ll、rr、cc,以空格隔开。
若 \mathrm{opt} = 0opt=0,表示将位于 [l, r][l,r] 的之间的数字都加 cc。
若 \mathrm{opt} = 1opt=1,表示将位于 [l, r][l,r] 的之间的数字都乘 cc。
若 \mathrm{opt} = 2opt=2,表示询问 a_rar 的值 \mathop{\mathrm{mod}} 10007mod10007(ll 和 cc 忽略)。
输出格式
对于每次询问,输出一行一个数字表示答案。
样例
样例输入
7
1 2 2 3 9 3 2
0 1 3 1
2 1 3 1
1 1 4 4
0 1 7 2
1 2 6 4
1 1 6 5
2 2 6 4
样例输出
3
100
数据范围与提示
对于 100\%100% 的数据,1 \leq n \leq 100000, -2^{31} \leq \mathrm{others}1≤n≤100000,−231≤others、\mathrm{ans} \leq 2^{31}-1ans≤231−1。
这道题要开比1e5大,要不然过不去,真实测试。。。
代码:
//#6283. 数列分块入门 7-区间乘法,区间加法,单点查询-要开1e6,mdzz
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxn=1e6+; int n,m;
int a[maxn],add[maxn],mul[maxn],pos[maxn];
const int mod=; void update_add(int l,int r,int c)
{
if(pos[l]==pos[r]){
for(int i=(pos[l]-)*m+;i<=pos[l]*m;i++){
a[i]=(a[i]*mul[pos[l]]%mod+add[pos[l]]+mod)%mod;
}
for(int i=l;i<=r;i++)
a[i]=(a[i]+c+mod)%mod;
mul[pos[l]]=;add[pos[l]]=;
}
else{
for(int i=(pos[l]-)*m+;i<=pos[l]*m;i++){
a[i]=(a[i]*mul[pos[l]]%mod+add[pos[l]]+mod)%mod;
}
for(int i=l;i<=pos[l]*m;i++)
a[i]=(a[i]+c+mod)%mod;
mul[pos[l]]=;add[pos[l]]=;
for(int i=pos[l]+;i<pos[r];i++){
add[i]=(add[i]+c+mod)%mod;
}
for(int i=(pos[r]-)*m+;i<=min(pos[r]*m,n);i++){
a[i]=(a[i]*mul[pos[r]]%mod+add[pos[r]]+mod)%mod;
}
for(int i=(pos[r]-)*m+;i<=r;i++)
a[i]=(a[i]+c+mod)%mod;
mul[pos[r]]=;add[pos[r]]=;
}
} void update_mul(int l,int r,int c)
{
if(pos[l]==pos[r]){
for(int i=(pos[l]-)*m+;i<=pos[l]*m;i++){
a[i]=(a[i]*mul[pos[l]]%mod+add[pos[l]]+mod)%mod;
}
for(int i=l;i<=r;i++)
a[i]=(a[i]*c+mod)%mod;
mul[pos[l]]=;add[pos[l]]=;
}
else{
for(int i=(pos[l]-)*m+;i<=pos[l]*m;i++){
a[i]=(a[i]*mul[pos[l]]%mod+add[pos[l]]+mod)%mod;
}
for(int i=l;i<=pos[l]*m;i++)
a[i]=(a[i]*c+mod)%mod;
mul[pos[l]]=;add[pos[l]]=;
for(int i=pos[l]+;i<pos[r];i++){
add[i]=(add[i]*c+mod)%mod;
mul[i]=(mul[i]*c+mod)%mod;
}
for(int i=(pos[r]-)*m+;i<=min(pos[r]*m,n);i++){
a[i]=(a[i]*mul[pos[r]]%mod+add[pos[r]]+mod)%mod;
}
for(int i=(pos[r]-)*m+;i<=r;i++)
a[i]=(a[i]*c+mod)%mod;
mul[pos[r]]=;add[pos[r]]=;
}
} int main()
{
scanf("%d",&n);
m=sqrt(n);
for(int i=;i<=n;i++){
scanf("%d",&a[i]);
a[i]%=mod;
pos[i]=(i-)/m+;
mul[pos[i]]=;
add[pos[i]]=;
}
for(int i=;i<=n;i++){
int op,l,r,c;
scanf("%d%d%d%d",&op,&l,&r,&c);
c%=mod;
if(op==){
update_add(l,r,c);
}
else if(op==){
update_mul(l,r,c);
}
else if(op==){
printf("%d\n",(a[r]*mul[pos[r]]+add[pos[r]])%mod);
}
}
} /*
10
1 3 4 2 5 7 11 3 5 1
0 1 5 1
1 1 7 2
2 3 9 1
0 4 8 1
1 1 5 2
1 3 5 2
2 5 7 1
1 3 5 2
2 2 3 2
2 3 4 5 5
23
80
56
*/
LOJ #6283. 数列分块入门 7-分块(区间乘法、区间加法、单点查询)的更多相关文章
- bzoj 1798: [Ahoi2009]Seq 维护序列seq 线段树 区间乘法区间加法 区间求和
1798: [Ahoi2009]Seq 维护序列seq Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://www.lydsy.com/JudgeO ...
- LOJ#6283. 数列分块入门 7
对于每个区间先乘在加,如果我修改的是部分的块,我就需要把现这个块的add和mul标记全部放下去,然后再更新. #include<map> #include<set> #incl ...
- LOJ.6281.数列分块入门5(分块 区间开方)
题目链接 int内的数(也不非得是int)最多开方4.5次就变成1了,所以还不是1就暴力,是1就直接跳过. #include <cmath> #include <cstdio> ...
- LibreOJ 6280 数列分块入门 4(分块区间加区间求和)
题解:分块的区间求和比起线段树来说实在是太好写了(当然,复杂度也高)但这也是没办法的事情嘛.总之50000的数据跑了75ms左右还是挺优越的. 比起单点询问来说,区间询问和也没有复杂多少,多开一个su ...
- LOJ.6284.数列分块入门8(分块)
题目链接 \(Description\) 给出一个长为n的数列,以及n个操作,操作涉及区间询问等于一个数c的元素,并将这个区间的所有元素改为c. \(Solution\) 模拟一些数据可以发现,询问后 ...
- LibreOJ 6281 数列分块入门 5(分块区间开方区间求和)
题解:区间开方emmm,这马上让我想起了当时写线段树的时候,很显然,对于一个在2^31次方以内的数,开方7-8次就差不多变成一了,所以我们对于每次开方,如果块中的所有数都为一了,那么开方也没有必要了. ...
- LibreOJ 6277 数列分块入门 1(分块)
题解:感谢hzwer学长和loj让本蒟蒻能够找到如此合适的入门题做. 这是一道非常标准的分块模板题,本来用打标记的线段树不知道要写多少行,但是分块只有这么几行,极其高妙. 代码如下: #include ...
- [Libre 6281] 数列分块入门 5 (分块)
水一道入门分块qwq 题面:传送门 开方基本暴力.. 如果某一个区间全部都开成1或0就打上标记全部跳过就行了 因为一个数开上个四五六次就是1了所以复杂度能过233~ code: //By Menteu ...
- LibreOJ 6278 数列分块入门 2(分块)
题解:非常高妙的分块,每个块对应一个桶,桶内元素全部sort过,加值时,对于零散块O(sqrt(n))暴力修改,然后暴力重构桶.对于大块直接整块加.查询时对于非完整块O(sqrt(n))暴力遍历.对 ...
随机推荐
- git代码冲突
如果系统中有一些配置文件在服务器上做了配置修改,然后后续开发又新添加一些配置项的时候, 在发布这个配置文件的时候,会发生代码冲突: error: Your local changes to the f ...
- String StrigBuffer StringBuilder 浅解
1.String是最基本的字符串类,用于表示字符串. 特点:对象内容不可变,但可以通过指向不同的对象来“表示”不同的内容. 使用场景:如果不涉及到内容改变,可以使用String. 注意:如果想将Str ...
- Linux修改用户密码
1. root修改自己 # passwd 2. root修改别人 # passwd oracle //修改oracle的密码
- Spring Security 集成CAS实现单点登录
参考:http://elim.iteye.com/blog/2270446 众所周知,Cas是对单点登录的一种实现.本文假设读者已经了解了Cas的原理及其使用,这些内容在本文将不会讨论.Cas有Ser ...
- [cerc2012][Gym100624C]20181013
题意:用元素符号表示字符串 题解:签到题 简单dp 难点在于把元素符号都改成小写qaq #include<cstdio> #include<cstdlib> #include& ...
- 【TYVJ】P1039 忠诚2
[算法]线段树 [注意]修改或查询区间时,若区间能包含某棵子树就立即返回,否则线段树就失去了意义. #include<cstdio> #include<algorithm> u ...
- windows下 nginx安装 使用
介绍 Nginx (engine x) 是一个高性能的HTTP和反向代理服务器. 反向代理(Reverse Proxy)方式是指以代理服务器来接受internet上的连接请求,然后将请求转发给内部网络 ...
- shell 文件内容替换 sed用法
sed 's/要被替换的字符串/新的字符串/g' sed 's/test/mytest/g' example-----在整行范围内把test替换为mytest. 如果没有g标记,则只有每行第一个匹配的 ...
- Java的9种基本数据类型的大小,以及他们的封装类
由于java程序是运行在虚拟机之上的,所以java的基本数据类型的大小是确定的,不会随着操作系统的位数的改变而改变. 在计算机中,存储的是0,1,0,1这样的二进制位,表示为bit,1Byte = 8 ...
- gpio子系统和pinctrl子系统(下)
情景分析 打算从两个角度来情景分析,先从bsp驱动工程师的角度,然后是驱动工程师的角度,下面以三星s3c6410 Pinctrl-samsung.c为例看看pinctrl输入参数的初始化过程(最开始的 ...