【分块】教主的魔法 @洛谷P2801/upcexam3138
时间限制: 1 Sec 内存限制: 128 MB
题目描述
教主最近学会了一种神奇的魔法,能够使人长高。于是他准备演示给XMYZ信息组每个英雄看。于是N个英雄们又一次聚集在了一起,这次他们排成了一列,被编号为1、2、……、N。
每个人的身高一开始都是不超过1000的正整数。教主的魔法每次可以把闭区间[L, R](1≤L≤R≤N)内的英雄的身高全部加上一个整数W。(虽然L=R时并不符合区间的书写规范,但我们可以认为是单独增加第L(R)个英雄的身高)
CYZ、光哥和ZJQ等人不信教主的邪,于是他们有时候会问WD闭区间 [L, R] 内有多少英雄身高大于等于C,以验证教主的魔法是否真的有效。
WD巨懒,于是他把这个回答的任务交给了你。
输入
第1行为两个整数N、Q。Q为问题数与教主的施法数总和。
第2行有N个正整数,第i个数代表第i个英雄的身高。
第3到第Q+2行每行有一个操作:
(1)若第一个字母为“M”,则紧接着有三个数字L、R、W。表示对闭区间 [L, R] 内所有英雄的身高加上W。
(2)若第一个字母为“A”,则紧接着有三个数字L、R、C。询问闭区间 [L, R] 内有多少英雄的身高大于等于C。
输出
对每个“A”询问输出一行,仅含一个整数,表示闭区间 [L, R] 内身高大于等于C的英雄数。
样例输入
5 3
1 2 3 4 5
A 1 5 4
M 3 5 1
A 1 5 4
样例输出
2
3
提示
原先5个英雄身高为1、2、3、4、5,此时[1, 5]间有2个英雄的身高大于等于4。教主施法后变为1、2、4、5、6,此时[1, 5]间有3个英雄的身高大于等于4。
对30%的数据,N≤1000,Q≤1000。
对100%的数据,N≤1000000,Q≤3000,1≤W≤1000,1≤C≤1,000,000,000
第一次写分块题
将数据分成sqrt(n)块,对每个块分别排序。
对于修改操作,区间内完整的块修改其add数组的值,不完整的块暴力修改在区间内的部分,然后对整个块排序。
对于查询操作,区间内完整的块二分查找C-add[i]在区间内的位置,然后用该位置减去块的首位置,不完整的块暴力统计。
#define FILE_PC() freopen("C:\\Users\\hz\\Desktop\\in.txt","r",stdin)
#include <algorithm>
#include <cmath>
#include <cstdio>
#include <cstdlib>
#include <cstring>
using namespace std;
const int maxn = 1000005;
const int maxdiv = 1005;
int a[maxn],b[maxn];
int l[maxdiv],r[maxdiv],add[maxdiv];
int lendiv,n,m;
int main() {
// FILE_PC();
scanf("%d%d",&n,&m);
for(int i=1; i<=n; i++) {
scanf("%d",a+i);
b[i] = a[i];
}
lendiv = (int)sqrt(n); //分块长度
int cntdiv = 0,flag = 1;
for(int i=1; i<=n; i++) {
if(cntdiv==0)l[++cntdiv] = i;//块的编号从1开始
if(!flag) {
l[++cntdiv] = i;
flag = 1;
}
if(i%lendiv==0) {
r[cntdiv] = i;
flag = 0;
}
}
r[cntdiv] = n; //最后一块的右区间
for(int i=1; i<=cntdiv; i++) {
sort(b+l[i],b+r[i]+1);
}//每个块里面的元素排序
for(int ca=0; ca<m; ca++) {
char s[20];
int ll,rr,cc;
scanf("%s%d%d%d",s,&ll,&rr,&cc);
if(s[0]=='M') {
int ld = (int)(lower_bound(l+1,l+cntdiv+1,ll)-l);
if(l[ld]!=ll) {
for(int i=ll; i<min(l[ld],rr+1); i++) {
a[i]+=cc;
}
for(int i=l[ld-1]; i<=r[ld-1]; i++) {
b[i] = a[i];
}
sort(b+l[ld-1],b+r[ld-1]+1);//重新对这个块排序
}
if(l[ld]<rr+1) {
int rd = (int)(lower_bound(r+1,r+cntdiv+1,rr)-r);
if(rr!=r[rd]) {
for(int i=max(ll,r[rd-1]+1); i<=rr; i++) {
a[i] += cc;
}
for(int i=l[rd]; i<=r[rd]; i++) {
b[i] = a[i];
}
sort(b+l[rd],b+r[rd]+1);//同上
}
for(int i=ld; i<=((rr==r[rd])?rd:rd-1); i++) { //完整的块
add[i]+=cc;
}
}
} else {
int ans = 0;
int ld = (int)(lower_bound(l+1,l+cntdiv+1,ll)-l);//给定的左区间落在哪一块
if(l[ld]!=ll) {
for(int i=ll; i<min(l[ld],rr+1); i++) { //暴力统计不完整的块
if(a[i]+add[ld-1]>=cc)ans++;
}
}
if(l[ld]<rr+1) {
int rd = (int)(lower_bound(r+1,r+cntdiv+1,rr)-r);//给定的右区间落在哪一块
if(rr!=r[rd]) {
for(int i=max(ll,r[rd-1]+1); i<=rr; i++) { //同上
if(a[i]+add[rd]>=cc)ans++;
}
}
for(int i=ld; i<=((rr==r[rd])?rd:rd-1); i++) { //统计每一个完整的块
ans += (int)(b+r[i]+1-lower_bound(b+l[i],b+r[i]+1,cc-add[i]));
}
}
printf("%d\n",ans);
}
}
return 0;
}
【分块】教主的魔法 @洛谷P2801/upcexam3138的更多相关文章
- BZOJ——3343: 教主的魔法 || 洛谷—— P2801 教主的魔法
http://www.lydsy.com/JudgeOnline/problem.php?id=3343 || https://www.luogu.org/problem/show?pid=280 ...
- 洛谷P2801 教主的魔法 [分块,二分答案]
题目传送门 教主的魔法 题目描述 教主最近学会了一种神奇的魔法,能够使人长高.于是他准备演示给XMYZ信息组每个英雄看.于是N个英雄们又一次聚集在了一起,这次他们排成了一列,被编号为1.2.…….N. ...
- 洛谷——P2801 教主的魔法(线段树or分块)
P2801 教主的魔法 (1) 若第一个字母为“M”,则紧接着有三个数字L.R.W.表示对闭区间 [L, R] 内所有英雄的身高加上W. (2) 若第一个字母为“A”,则紧接着有三个数字L.R.C.询 ...
- 洛谷 P2801 教主的魔法 解题报告
P2801 教主的魔法 题目描述 教主最近学会了一种神奇的魔法,能够使人长高.于是他准备演示给XMYZ信息组每个英雄看.于是N个英雄们又一次聚集在了一起,这次他们排成了一列,被编号为1.2.--.N. ...
- 洛谷 P2801 教主的魔法
题目描述 教主最近学会了一种神奇的魔法,能够使人长高.于是他准备演示给XMYZ信息组每个英雄看.于是N个英雄们又一次聚集在了一起,这次他们排成了一列,被编号为1.2.…….N. 每个人的身高一开始都是 ...
- 【算法】分块——教主的魔法&不勤劳的图书管理员
由不勤劳的图书管理员带入了分块的坑,深深地被其暴力与优雅所征服.分块的实质就是将暴力块状封装起来,一整块的部分就一整块处理,零碎的部分就怎么暴力怎么来.因为分块大小的原因,限制了零碎部分数据的数量级, ...
- 洛谷P2801 教主的魔法 分块
正解:分块 解题报告: 哇之前的坑还没填完就又写新博客? 不管不管,之前欠的两三篇题解大概圣诞节之前会再仔细想想然后重新写下题解趴,确实还挺难的感觉没有很好的理解呢QAQ还是太囫囵吞枣不求甚解了,这样 ...
- [洛谷P2801]教主的魔法
题目大意:有$n$个数,$q$个操作.两种操作: $M\;l\;r\;w:$把$[l,r]$所有数加上$w$ $A\;l\;r\;c:$查询$[l,r]$内大于等于$c$的元素的个数. 题解:分块,对 ...
- 洛谷 P2801 教主的魔法 题解
题面 刚看到这道题的时候用了个树状数组优化前缀和差分的常数优化竟然AC了?(这数据也太水了吧~) 本人做的第一道分块题,调试了好久好久,最后竟然没想到二分上还会出错!(一定要注意)仅此纪念: #inc ...
随机推荐
- 第k个互质数(二分 + 容斥)
描述两个数的a,b的gcd为1,即a,b互质,现在给你一个数m,你知道与它互质的第k个数是多少吗?与m互质的数按照升序排列. 输入 输入m ,k (1<=m<=1000000;1<= ...
- mysql四大特性与四种隔离级别
四大特性 1:原子性.事务是一个不可分割的整体,事务开始的操作,要么全部执行,要么全部不执行. 2:隔离性.同一时间,只允许一个事务请求同一组数据.不同的事务彼此之间没有干扰. 3:一致性.事务开始前 ...
- 【bzoj2023/1630】[Usaco2005 Nov]Ant Counting 数蚂蚁 dp
题解: 水题 f[i][j] 前i种用了j个,前缀和优化就可以了
- mybatis拦截器处理
1.自定义注释 package com.hsfw.backyard.biz.security.authority; import java.lang.annotation.*; /** * 数据权限过 ...
- Python 携程
一.协程 1.又称微线程,纤程.英文名Coroutine.一句话说明什么是协程:协程是一种用户态的轻量级线程(相当于操作系统不知道它的存在,是用户控制的). 2.协程拥有自己的寄存器上下文和栈(代码的 ...
- 51Nod1766 树上的最远点对 ST表 LCA 线段树
原文链接https://www.cnblogs.com/zhouzhendong/p/51Nod1766.html 题目传送门 - 51Nod1766 题意 n个点被n-1条边连接成了一颗树,给出a~ ...
- IDEA链接mySql问题 : You have an error in your SQL syntax : 'OPTION SQL_SELECT_LIMIT=1000' (or 'OPTION SQL_SELECT_LIMIT=DEFAULT')
IDEA控制台错误信息: check the manual that corresponds to your MySQL server version for the right Code: 1064 ...
- Python之抽象类、抽象方法
抽象类中只能有抽象方法,子类继承抽象类时,不能通过实例化使用其抽象方法,必须实现该方法. Python2 class CopyBase(object): def save(self): raise N ...
- Machine Learning 算法可视化实现1 - 线性回归
一.原理和概念 1.回归 回归最简单的定义是,给出一个点集D,用一个函数去拟合这个点集.而且使得点集与拟合函数间的误差最小,假设这个函数曲线是一条直线,那就被称为线性回归:假设曲线是一条二次曲线,就被 ...
- 队列queue实现线程的消费者和生产者
import threading import queue import random import time qq = queue.Queue(4) #实例化一个队列,因为是一个进程的线程,所以共资 ...