分块入门题,不错的,建议大家做一做

开始学习

先看一下数列分块入门 2

这道题想让我们求区间[l,r]>=c的个数,然后我们可以看到“数列分块入门 2”是求区间[l,r]<c(忽略平方)的个数,即求c在区间[l,r]的排名。所以我们可以每一次查询c的排名,然后用区间长度减c的排名就可以达到答案了呢QAQ。

那么题目就被转移成了求区间[l,r]中c的排名


一看题目,咦,求[l,r]区间c的排名,马上就可以想到平衡树啦,可是平衡树这么难写,而且还不支持区间加,那怎么办? 分块!

首先,我们一个一个步骤分着来看:

(1).操作 操作和普通分块的操作一模一样,类个tag,我们先不理。

(2).查询 如果一般查询排名,我们可以将这一段数截出来,排序,然后二分查找(lower_bound操作)就ok了。那我们可不可以先将每一个块的数都排好序,然后查找每一个块的时候就直接二分呢? ofcause,当然可以!那零零散散的剩下的非整块的数就暴力找就可以了。

(3).排序操作 我们查询需要先排序,那么我们排序也成为了一个操作。我们另外开一个数组ve[i](ve[i]为vector,i表示第i个块)来存每个块排序的数据(这里我使用了vector)。到时候查询的时候直接二分ve[i]就可以了。那每一次零散修改操作都会修改数值(整块操作还是直接加tag),那么我们要对有修改数值的那个块重新进行块内排序。(额,好慢啊)

所以这道题的复杂度是O(msqrt(nlogn))

那么我们这个分块的基本思想就是这样了。还有很多小细节可以看看我的代码:

//P2801 教主的魔法
#include<map>
#include<set>
#include<cmath>
#include<stack>
#include<queue>
#include<cstdio>
#include<vector>
#include<cstring>
#include<cstdlib>
#include<iostream>
#include<algorithm>
#define ll long long
using namespace std;
int n,m,blo;
int v[1000005],bl[1000005],atag[1000005];
vector<int>ve[10001];
void reset(int x) //排序操作
{
ve[x].clear(); //先初始化
for(int i=(x-1)*blo+1;i<=min(x*blo,n);i++)
ve[x].push_back(v[i]); //将整个块的数值重新压进vector里
sort(ve[x].begin(),ve[x].end()); //块内排序
}
ll read() //快读
{
ll x=0,f=1;char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();}
return x*f;
}
void add(int a,int b,int c) //修改操作
{
for(int i=a;i<=min(bl[a]*blo,b);i++) v[i]+=c; //零散操作
reset(bl[a]); //重新块内排序
if(bl[a]!=bl[b])
{
for(int i=(bl[b]-1)*blo+1;i<=b;i++) v[i]+=c;//零散操作
reset(bl[b]); //重新块内排序
}
for(int i=bl[a]+1;i<=bl[b]-1;i++)
atag[i]+=c;
}
int query(int a,int b,int c) //查询操作
{
int ans=0;//排名
for(int i=a;i<=min(bl[a]*blo,b);i++)//零散暴力查询
{
if(v[i]+atag[bl[a]]<c)
ans++;
}
if(bl[a]!=bl[b])
{
for(int i=(bl[b]-1)*blo+1;i<=b;i++) //零散暴力查询
{
if(v[i]+atag[bl[b]]<c)
ans++;
}
}
for(int i=bl[a]+1;i<=bl[b]-1;i++) //整块查询
{
int x=c-atag[i]; //注意:这里要先剪去这个块的atag,因为这整个块的所以数值都应该加了atag,所以我们二分找的数也要先剪atag
ans+=lower_bound(ve[i].begin(),ve[i].end(),x)-ve[i].begin(); //注意:这里lower_bound操作返回的是地址,所以要想知道块内排名,要减块头的地址
} //不会用lower_bound可以直接写二分
return ans;
}
int main()
{
n=read();m=read();blo=sqrt(n);
for(int i=1;i<=n;i++)v[i]=read();
for(int i=1;i<=n;i++)bl[i]=(i-1)/blo+1,ve[bl[i]].push_back(v[i]); //初始化ve[i]数组
for(int i=1;i<=bl[n];i++)
sort(ve[i].begin(),ve[i].end()); //排序
for(int i=1;i<=m;i++)
{
char f;cin>>f;
int a=read(),b=read(),c=read();
if(f=='M')add(a,b,c);
if(f=='A')printf("%d\n",b-a+1-query(a,b,c));
}
return 0;
}

广告

如果需要更系统的学习分块可以看我的blog,内容更详细,分解更到位qwq

分块学习blog

题解 P2801 【教主的魔法】的更多相关文章

  1. P2801 教主的魔法(分块)

    P2801 教主的魔法 区间加法,区间查询 显然就是分块辣 维护一个按块排好序的数组. 每次修改依然是整块打标记,零散块暴力.蓝后对零散块重新排序. 询问时整块二分,零散块暴力就好辣 注意细节挺多和边 ...

  2. 洛谷 P2801 教主的魔法 解题报告

    P2801 教主的魔法 题目描述 教主最近学会了一种神奇的魔法,能够使人长高.于是他准备演示给XMYZ信息组每个英雄看.于是N个英雄们又一次聚集在了一起,这次他们排成了一列,被编号为1.2.--.N. ...

  3. 洛谷——P2801 教主的魔法(线段树or分块)

    P2801 教主的魔法 (1) 若第一个字母为“M”,则紧接着有三个数字L.R.W.表示对闭区间 [L, R] 内所有英雄的身高加上W. (2) 若第一个字母为“A”,则紧接着有三个数字L.R.C.询 ...

  4. P2801 教主的魔法 (线段树)

    题目 P2801 教主的魔法 解析 成天做水题 线段树,第一问区间加很简单 第二问可以维护一个区间最大值和一个区间最小值,若C小于等于区间最小值,就加上区间长度,若C大于区间最大值,就加0 ps:求教 ...

  5. 洛谷P2801 教主的魔法 [分块,二分答案]

    题目传送门 教主的魔法 题目描述 教主最近学会了一种神奇的魔法,能够使人长高.于是他准备演示给XMYZ信息组每个英雄看.于是N个英雄们又一次聚集在了一起,这次他们排成了一列,被编号为1.2.…….N. ...

  6. luogu P2801 教主的魔法

    题目描述 教主最近学会了一种神奇的魔法,能够使人长高.于是他准备演示给XMYZ信息组每个英雄看.于是N个英雄们又一次聚集在了一起,这次他们排成了一列,被编号为1.2.…….N. 每个人的身高一开始都是 ...

  7. 洛谷 P2801 教主的魔法

    题目描述 教主最近学会了一种神奇的魔法,能够使人长高.于是他准备演示给XMYZ信息组每个英雄看.于是N个英雄们又一次聚集在了一起,这次他们排成了一列,被编号为1.2.…….N. 每个人的身高一开始都是 ...

  8. BZOJ——3343: 教主的魔法 || 洛谷—— P2801 教主的魔法

    http://www.lydsy.com/JudgeOnline/problem.php?id=3343  ||  https://www.luogu.org/problem/show?pid=280 ...

  9. 洛谷 P2801 教主的魔法 题解

    题面 刚看到这道题的时候用了个树状数组优化前缀和差分的常数优化竟然AC了?(这数据也太水了吧~) 本人做的第一道分块题,调试了好久好久,最后竟然没想到二分上还会出错!(一定要注意)仅此纪念: #inc ...

  10. 洛谷P2801 教主的魔法 分块

    正解:分块 解题报告: 哇之前的坑还没填完就又写新博客? 不管不管,之前欠的两三篇题解大概圣诞节之前会再仔细想想然后重新写下题解趴,确实还挺难的感觉没有很好的理解呢QAQ还是太囫囵吞枣不求甚解了,这样 ...

随机推荐

  1. 吴裕雄 python 神经网络——TensorFlow 数据集基本使用方法

    import tempfile import tensorflow as tf input_data = [1, 2, 3, 5, 8] dataset = tf.data.Dataset.from_ ...

  2. mxgraph初体验

    最近公司让学习了mxgraph,简单总结一下 (1)mxGraph学习路径 1)API:http://jgraph.github.io/mxgraph/docs/js-api/files/index- ...

  3. PCSearch

    1.hinstance:GetModuleHandle(NULL) 2.窗口直角: 方法1:在Oncreate函数中添加以下代码,然而这种方法会导致窗口阴影无效 LONG styleValue = : ...

  4. 关于calendar修改前的代码和修改后的代码

    Java编写的日历,输入年月,输出这个月的日期与星期 修改前的代码: import java.io.BufferedReader; import java.io.IOException; import ...

  5. stm32 串口发送字符串丢失第一个字节

    使用stm32f103调试串口通讯时,上电后发送的字符串的第一个字节丢失. 发送数据GpuSend("SPG(2);"); 接收端收到的数据为:PG(2);,第一个字符丢失. 出现 ...

  6. SpringBoot与Jpa入门

    一.JPA简介 目前JPA主要实现由hibernate和openJPA等. Spring Data JPA 是Spring Data 的一个子项目,它通过提供基于JPA的Repository极大了减少 ...

  7. day1-3js代码执行特性

    Js代码执行特性 js中变量声明都会提升到脚本的第一行(注意不是定义,只是声明) 函数变量声明也会提升到前面(是整个函数!)(变量最前,函数其后) 注:在执行js代码前,先把所有变量声明,函数提升至前 ...

  8. 用IDEA创建项目时没有Spring类型的项目模板

    使用的版本:2019.2.2 Community 解决方法:File=>Setting=>Plugins=>搜索Spring,安装Spring Assistant=>重启IDE ...

  9. day 10 作业

    # 2.写函数,接收n个数字,求这些参数数字的和. def sum_func(*args): total = 0 for i in args: total += i return total prin ...

  10. ZeroTier 局域网组建工具

    无公网IP通过ZeroTier实现内网穿透 需求:想要在公司访问家里内网NAS,或是在家里访问公司服务 有固定的公网IP或动态的公网IP:常见的方案动态域名解析做端口转发方式等 无公网IP:常见的实现 ...