LOJ数列分块入门九题(上)
一转眼,已经有整整一年没有在博客园写博客了。去洛谷写了几篇(How time flys.
最近突然想起其实我不太擅长分块算法,又想起去年暑假有位同学同我提起过LOJ的数列分块九题,说来惭愧,打了这么久题今天才打算做这个系列,甚至连LOJ的账号都没有注册。。。于是今天花费了半天时间敲题,但是更尴尬的是只打了前四题,分块细节真的是相当繁琐。
分块是一种优雅的暴力,时间复杂度为根号n量级,虽然同线段树树状数组之类的logn量级的复杂度无法相提并论,但是在大多数情况下,分块都可以比较好地替代线段树。其代码量较少且逻辑相较线段树而言更清晰尤其是可以进行块内二分之类的操作,从而有的时候比线段树更有优势。
分块,顾名思义,就是将数列分为多块进行考虑,在进行区间修改或者区间查询操作时,使用分块算法,我们可以直接处理区间中的块,再处理零散块。我们可以将数列划分为根号n个块,从而将时间复杂度降低至根号n。
#6277. 数列分块入门 1 - 题目 - LibreOJ (loj.ac)
区间加法,单点查值,分块水题(变量名想半天)
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxn=(5e4)+5;
ll a[maxn];
ll l[maxn];
ll r[maxn];
ll bel[maxn];
ll lazy_add[maxn];
void build(ll n){
ll block=sqrt(n);
ll num=n/block;
if(n%block)num++;
for(int i=1;i<=num;i++){
l[i]=(i-1)*block+1;
r[i]=min(n,i*block);
lazy_add[i]=0;
}
for(int i=1;i<=n;i++){
bel[i]=(i-1)/block+1;
}
}
void add(ll L,ll rr,ll c){
ll s=bel[L]+1;
ll t=bel[rr]-1;
for(int i=L;i<=min(rr,r[s-1]);i++){
a[i]+=c;
}
if(bel[L]==bel[rr])return;
for(int i=l[t+1];i<=rr;i++){
a[i]+=c;
}
for(int i=s;i<=t;i++){
lazy_add[i]+=c;
}
}
void calc(ll x){
ll p=bel[x];
for(int i=l[p];i<=r[p];i++){
a[i]+=lazy_add[p];
}
lazy_add[p]=0;
cout<<a[x]<<endl;
}
int main(){
ll n;
cin>>n;
build(n);
for(int i=1;i<=n;i++)cin>>a[i];
for(int i=1;i<=n;i++){
ll opt,L,rr,c;
cin>>opt>>L>>rr>>c;
if(!opt){
add(L,rr,c);
}
else{
calc(rr);
}
}
return 0;
}
#6278. 数列分块入门 2 - 题目 - LibreOJ (loj.ac)
块内二分即可(线段树不行),注意边角的处理
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxn=(5e4)+5;
int L[maxn],R[maxn];
int bel[maxn];
ll a[maxn];
ll b[maxn];
ll tmp[maxn];
ll tag[maxn];
void build(int n){
int block=sqrt(n);
int num=n/block;
if(n%block)num++;
for(int i=1;i<=num;i++){
L[i]=(i-1)*block+1;
R[i]=min(n,i*block);
sort(a+L[i],a+R[i]+1);
for(int j=L[i];j<=R[i];j++){
bel[j]=i;
}
}
}
void update(int l,int r,ll c){
if(bel[l]==bel[r]){
for(int i=l;i<=r;i++)b[i]+=c;
for(int i=L[bel[l]];i<=R[bel[l]];i++){
tmp[i]=b[i];
}
sort(tmp+L[bel[l]],tmp+R[bel[l]]+1);
for(int i=L[bel[l]];i<=R[bel[l]];i++){
a[i]=tmp[i];
}
}
else{
for(int i=l;i<=R[bel[l]];i++){
b[i]+=c;
}
for(int i=L[bel[l]];i<=R[bel[l]];i++){
tmp[i]=b[i];
}
sort(tmp+L[bel[l]],tmp+R[bel[l]]+1);
for(int i=L[bel[l]];i<=R[bel[l]];i++){
a[i]=tmp[i];
}
for(int i=L[bel[r]];i<=r;i++){
b[i]+=c;
}
for(int i=L[bel[r]];i<=R[bel[r]];i++){
tmp[i]=b[i];
}
sort(tmp+L[bel[r]],tmp+R[bel[r]]+1);
for(int i=L[bel[r]];i<=R[bel[r]];i++){
a[i]=tmp[i];
}
for(int i=bel[l]+1;i<bel[r];i++){
tag[i]+=c;
}
}
}
void calc(int l,int r,ll c){
ll ans=0;
if(bel[l]==bel[r]){
for(int i=l;i<=r;i++){
if(b[i]+tag[bel[l]]<c)ans++;
}
}
else{
for(int i=l;i<=R[bel[l]];i++){
if(b[i]+tag[bel[l]]<c)ans++;
}
for(int i=L[bel[r]];i<=r;i++){
if(b[i]+tag[bel[r]]<c)ans++;
}
for(int i=bel[l]+1;i<bel[r];i++){
ans+=lower_bound(a+L[i],a+R[i]+1,c-tag[i])-a-L[i];
}
}
cout<<ans<<endl;
}
int main(){
int n;
cin>>n;
for(int i=1;i<=n;i++)cin>>a[i],b[i]=a[i];
build(n);
for(int i=1;i<=n;i++){
ll opt,l,r,c;
cin>>opt>>l>>r>>c;
if(!opt){
update(l,r,c);
}
else{
calc(l,r,c*c);
}
}
return 0;
}
#6279. 数列分块入门 3 - 题目 - LibreOJ (loj.ac)
前一题修改一下即可
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxn=(1e5)+5;
int L[maxn],R[maxn];
int bel[maxn];
ll a[maxn];
ll b[maxn];
ll tmp[maxn];
ll tag[maxn];
void build(int n){
int block=sqrt(n);
int num=n/block;
if(n%block)num++;
for(int i=1;i<=num;i++){
L[i]=(i-1)*block+1;
R[i]=min(n,i*block);
sort(a+L[i],a+R[i]+1);
for(int j=L[i];j<=R[i];j++){
bel[j]=i;
}
}
}
void update(int l,int r,ll c){
if(bel[l]==bel[r]){
for(int i=l;i<=r;i++)b[i]+=c;
for(int i=L[bel[l]];i<=R[bel[l]];i++){
tmp[i]=b[i];
}
sort(tmp+L[bel[l]],tmp+R[bel[l]]+1);
for(int i=L[bel[l]];i<=R[bel[l]];i++){
a[i]=tmp[i];
}
}
else{
for(int i=l;i<=R[bel[l]];i++){
b[i]+=c;
}
for(int i=L[bel[l]];i<=R[bel[l]];i++){
tmp[i]=b[i];
}
sort(tmp+L[bel[l]],tmp+R[bel[l]]+1);
for(int i=L[bel[l]];i<=R[bel[l]];i++){
a[i]=tmp[i];
}
for(int i=L[bel[r]];i<=r;i++){
b[i]+=c;
}
for(int i=L[bel[r]];i<=R[bel[r]];i++){
tmp[i]=b[i];
}
sort(tmp+L[bel[r]],tmp+R[bel[r]]+1);
for(int i=L[bel[r]];i<=R[bel[r]];i++){
a[i]=tmp[i];
}
for(int i=bel[l]+1;i<bel[r];i++){
tag[i]+=c;
}
}
}
void calc(int l,int r,ll c){
ll ans=-1e9;
bool fg=0;
if(bel[l]==bel[r]){
for(int i=l;i<=r;i++){
if(b[i]+tag[bel[l]]<c)ans=max(ans,b[i]+tag[bel[l]]),fg=1;
}
}
else{
for(int i=l;i<=R[bel[l]];i++){
if(b[i]+tag[bel[l]]<c)ans=max(ans,b[i]+tag[bel[l]]),fg=1;
}
for(int i=L[bel[r]];i<=r;i++){
if(b[i]+tag[bel[r]]<c)ans=max(ans,b[i]+tag[bel[r]]),fg=1;
}
for(int i=bel[l]+1;i<bel[r];i++){
int p=lower_bound(a+L[i],a+R[i]+1,c-tag[i])-a-1;
if(a[p]+tag[i]<c)ans=max(ans,a[p]+tag[i]),fg=1;
}
}
if(fg)cout<<ans<<endl;
else cout<<-1<<endl;
}
int main(){
int n;
cin>>n;
for(int i=1;i<=n;i++)cin>>a[i],b[i]=a[i];
build(n);
for(int i=1;i<=n;i++){
ll opt,l,r,c;
cin>>opt>>l>>r>>c;
if(!opt){
update(l,r,c);
}
else{
calc(l,r,c);
}
}
return 0;
}
#6280. 数列分块入门 4 - 题目 - LibreOJ (loj.ac)
区间加法,区间求和(经典线段树,不过也可以分块)
#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int maxn=(1e5)+5;
ll a[maxn];
int L[maxn],R[maxn];
int bel[maxn];
ll siz[maxn];
ll tag[maxn];
void build(int n){
int block=sqrt(n);
int num=n/block;
if(n%block)num++;
for(int i=1;i<=num;i++){
L[i]=(i-1)*block+1;
R[i]=min(n,i*block);
for(int j=L[i];j<=R[i];j++){
bel[j]=i;
siz[i]+=a[j];
}
}
}
void update(int l,int r,ll c){
if(bel[l]==bel[r]){
for(int i=l;i<=r;i++){
a[i]+=c;
siz[bel[l]]+=c;
}
}
else{
for(int i=l;i<=R[bel[l]];i++){
a[i]+=c;
siz[bel[l]]+=c;
}
for(int i=L[bel[r]];i<=r;i++){
a[i]+=c;
siz[bel[r]]+=c;
}
for(int i=bel[l]+1;i<bel[r];i++){
siz[i]+=(R[i]-L[i]+1)*c;
tag[i]+=c;
}
}
}
void calc(int l,int r,ll c){
ll ans=0;
if(bel[l]==bel[r]){
for(int i=l;i<=r;i++){
ans=(ans+a[i]+tag[bel[l]])%c;
}
}
else{
for(int i=l;i<=R[bel[l]];i++){
ans=(ans+a[i]+tag[bel[l]])%c;
}
for(int i=L[bel[r]];i<=r;i++){
ans=(ans+a[i]+tag[bel[r]])%c;
}
for(int i=bel[l]+1;i<bel[r];i++){
ans=(ans+siz[i])%c;
}
}
cout<<ans%c<<endl;
}
int main(){
int n;
cin>>n;
for(int i=1;i<=n;i++)cin>>a[i];
build(n);
for(int i=1;i<=n;i++){
ll opt,l,r,c;
cin>>opt>>l>>r>>c;
if(!opt){
update(l,r,c);
}
else calc(l,r,c+1);
}
return 0;
}
LOJ数列分块入门九题(上)的更多相关文章
- LOJ 数列分块入门 9 题解题报告
LOJ 数列分块入门 9 题解题报告 \(\text{By DaiRuiChen007}\) I. 数列分块入门 1 题目大意 \(\text{Link}\) 维护一个长度为 \(n\) 的序列,支持 ...
- 数列分块入门九题(三):LOJ6283~6285
Preface 最后一题我一直觉得用莫队是最好的. 数列分块入门 7--区间乘法,区间加法,单点询问 还是很简单的吧,比起数列分块入门 7就多了个区间乘. 类似于线段树,由于乘法的优先级高于加法,因此 ...
- 数列分块入门九题(二):LOJ6280~6282
Preface 个人感觉这中间的三题是最水的没有之一 数列分块入门 4--区间加法,区间求和 这个也是很多数据结构完爆的题目线段树入门题,但是练分块我们就要写吗 修改还是与之前类似,只不过我们要维护每 ...
- 数列分块入门九题(一):LOJ6277~6279
Preface 分块,一个神奇的暴力算法.可以把很多\(O(n^2)\)的数据结构题的暴力优化到常数极小的\(O(n\sqrt n)\).当一些毒瘤题无法用线段树,主席树,平衡树,树状数组...... ...
- [Loj] 数列分块入门 1 - 9
数列分块入门 1 https://loj.ac/problem/6277 区间加 + 单点查询 #include <iostream> #include <cstdio> #i ...
- loj 数列分块入门 5 7 8
5 题意 给出一个长为\(n\)的数列,以及\(n\)个操作,操作涉及区间开方,区间求和. 思路 用\(tag\)记录这一块是否已全为\(1\). 除分块外,还可用 树状数组+并查集(链表) 或者 线 ...
- loj 数列分块入门 6 9(区间众数)
6 题意 给出一个长为\(n\)的数列,以及\(n\)个操作,操作涉及单点插入,单点询问,数据随机生成. 题解 参考:http://hzwer.com/8053.html 每个块内用一个\(vecto ...
- LOJ 数列分块入门系列
目录 1.区间加+单点查 每个块维护tag,散的暴力改. code: #include<bits/stdc++.h> using namespace std; const int maxn ...
- LOJ6285 数列分块入门9(分块)
昨天对着代码看了一晚上 然后今天终于在loj上过了 数列分块入门9题撒花★,°:.☆( ̄▽ ̄)/$:.°★ . 然后相当玄学 块的大小调成\(\sqrt{n}\)会TLE,改成150就过了 啧 然后就 ...
- LOJ——#6277. 数列分块入门 1
~~推荐播客~~ 「分块」数列分块入门1 – 9 by hzwer 浅谈基础根号算法——分块 博主蒟蒻,有缘人可直接观摩以上大佬的博客... #6277. 数列分块入门 1 题目大意: 给出一个长为 ...
随机推荐
- 【uni-app】第2节HBuilderX未检测到手机问题(今天终于找到解决办法了,亲测可以)
1.[问题描述]项目想在真机调试,开发者选项已开启USB调试,但是HBuilderX未检测到手机或模拟器 2.[解决办法] 2.1 找到HBuilderX安装目 ...
- vue-用户管理系统
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- 备份Cisco交换机设备配置
需求: 备份网络核心设备配置 工具: 1.3CDaemon软件,用于配置TFTP服务器 链接:http://www.china-ccie.com/download/3CDaemon/3CDaemon. ...
- 前端vue的JsPDF html2canvas 生成pdf并以文件流形式上传到后端(转载)
原文地址 1.首先在文件内引入htmlToPdf.js这里代码引入了html2canvas和jspdf//需要 npm i html2Canvas 和 npm i jspdf 在这里将getPdf 这 ...
- response status is 500 https://localhost:7129/swagger/v1/swagger.json
SwaggerGeneratorException: Conflicting method/path combination "GET Test" for actions - To ...
- vue element-ui form验证中自定义验证方式通过不返回true问题
项目中使用了element-ui的form验证,自定义了手机号的验证规则,验证不通过的时候定义了callback()扔出错误.但是忘了写通过的callback().导致form验证通过拿不到返回的va ...
- maven 引入了jar包,但却不能使用jar包里类
无报错,但是就是 无法 使用 lombok 的类. 发现classpath 里面也的确没有lombok jar包. 最后把json 的 version 属性加上 就正常了. 所以 结论: 不加vers ...
- Finance财务软件(月度结转专题)
支持按模板结转 默认结转模板 1.结转收入 借: 6001 主营业务收入 6051 其他业务收入 6301 营业外收入 贷: 4103 本年利润 2.结转成本.费用和税金 借: 4103 本年利润 贷 ...
- webstrom配置ES6 file watch没有生效
参考了文章https://www.cnblogs.com/kiimi/p/8663467.html设置后,依然没有看到编译后的文件,然后设置了输出文件夹为当前项目,再重启webstrom就看到了--
- Java流程控制之Scanner的进阶使用
Scanner的进阶使用 import java.util.Scanner; public class Demo04 { public static void main(String[] args) ...