loj6278 数列分块入门题2

题意:支持区间加,询问区间中元素排名

维护两个域。一个域维护原序列,一个域维护快内排序序列。

每次修改后更新快内排序序列。

修改时O(sqrt(n)log(sqrt(n)))

询问时O(sqrt(n)log(sqrt(n)))

大概是这个量级吧

 1 #include <bits/stdc++.h>
 2 using namespace std;
 3
 4 int bl[100005],bla,a[100005],a0[100005],b[100005],bi[100005],n;
 5 int t1,t2,t3,t4;
 6
 7 void init() {
 8     bla=sqrt(n);
 9     for (int i=1;i<=n;i++)
10         bl[i]=(i-1)/bla+1,
11         bi[i]=(i-1)%bla+1;
12     for (int l=1;l<=n;l+=bla) {
13         for(int i=bl[l]*bla-bla+1;i<=bl[l]*bla;i++)
14             a[i]=a0[i];
15         sort(a+bl[l]*bla-bla+1,a+bl[l]*bla+1);
16     }
17 }
18
19 void add(int l,int r,int c){
20     if(bl[l]==bl[r]) {
21         for(int i=l;i<=r;i++)
22             a0[i]+=c;
23         for(int i=bl[l]*bla-bla+1;i<=bl[l]*bla;i++)
24             a[i]=a0[i];
25         sort(a+bl[l]*bla-bla+1,a+bl[l]*bla+1);
26     }
27     else {
28         for(int i=l;i<=bl[l]*bla;i++)
29             a0[i]+=c;
30         for(int i=bl[l]*bla-bla+1;i<=bl[l]*bla;i++)
31             a[i]=a0[i];
32         sort(a+bl[l]*bla-bla+1,a+bl[l]*bla+1);
33         for(int i=bl[r]*bla-bla+1;i<=r;i++)
34             a0[i]+=c;
35         for(int i=bl[r]*bla-bla+1;i<=bl[r]*bla;i++)
36             a[i]=a0[i];
37         sort(a+bl[r]*bla-bla+1,a+bl[r]*bla+1);
38         for(int i=bl[l]+1;i<bl[r];i++)
39             b[i]+=c;
40     }
41 }
42
43 int query(int l,int r,long long c){
44     int ans=0;
45     if(bl[l]==bl[r]) {
46         for(int i=l;i<=r;i++)
47             ans+=(a0[i]<c-b[bl[l]]);
48     }
49     else {
50         for(int i=l;i<=bl[l]*bla;i++)
51             ans+=(a0[i]<c-b[bl[l]]);
52         for(int i=bl[r]*bla-bla+1;i<=r;i++)
53             ans+=(a0[i]<c-b[bl[r]]);
54         for(int i=bl[l]+1;i<bl[r];i++)
55             ans+=lower_bound(a+i*bla-bla+1,a+i*bla+1,c-b[i])-(a+i*bla-bla+1);
56     }
57     return ans;
58 }
59
60 void printblock(){
61     /*for(int i=1;i<=n/bla+(n%bla)?1:0;i++) {
62         printf("block %d : %d\n",i,b[i]);
63         printf(" a0: ");
64         for(int j=1;j<=bla;j++) printf("%d ",a0[i*bla-bla+j]);
65         printf("\n");
66         printf(" a:  ");
67         for(int j=1;j<=bla;j++) printf("%d ",a[i*bla-bla+j]);
68         printf("\n");
69     }*/
70 }
71
72 int main(){
73     ios::sync_with_stdio(false);
74     cin>>n;
75     for(int i=1;i<=n;i++)
76         cin>>a0[i];
77     init();
78     printblock();
79     for(int i=1;i<=n;i++) {
80         cin>>t1>>t2>>t3>>t4;
81         if(t1==0){
82             add(t2,t3,t4);
83         }
84         else {
85             printf("%d\n",query(t2,t3,t4*t4));
86         }
87         printblock();
88     }
89 }

原文地址:https://www.cnblogs.com/mollnn/p/8439927.html

时间: 2024-10-31 01:03:23

loj6278 数列分块入门题2的相关文章

loj6277 数列分块入门题1

裸题分块. 1 #include <bits/stdc++.h> 2 using namespace std; 3 4 int a[100005],b[10005],n,m,t1,t2,t3,t4,sq; 5 6 int main(){ 7 ios::sync_with_stdio(false); 8 cin>>n; 9 sq=(int)sqrt(n); 10 for(int i=1;i<=n;i++) cin>>a[i]; 11 for(int i=1;i<

分块入门题

分块入门题-(摘自黄学长的blog) 给出一个长为n的数列,以及n个操作,操作涉及区间加法,询问区间内小于某个值x的前驱(比其小的最大元素). n<=100000其实是为了区分暴力和一些常数较大的写法. 接着第二题的解法,其实只要把块内查询的二分稍作修改即可. 不过这题其实想表达:可以在块内维护其它结构使其更具有拓展性,比如放一个 set ,这样如果还有插入.删除元素的操作,会更加的方便. 分块的调试检测技巧: 可以生成一些大数据,然后用两份分块大小不同的代码来对拍,还可以根据运行时间尝试调整分

LOJ#6284. 数列分块入门 8

#6284. 数列分块入门 8 内存限制:256 MiB时间限制:500 ms标准输入输出 题目类型:传统评测方式:文本比较 上传者: hzwer 提交提交记录统计讨论 1 测试数据 题目描述 给出一个长为 nnn 的数列,以及 nnn 个操作,操作涉及区间询问等于一个数 ccc 的元素,并将这个区间的所有元素改为 ccc. 输入格式 第一行输入一个数字 nnn. 第二行输入 nnn 个数字,第 i 个数字为 aia_ia?i??,以空格隔开. 接下来输入 nnn 行询问,每行输入三个数字 ll

数列分块入门

分块是 莫队 算法的前置知识,也是一种十分 暴力 的数据结构. 分块的核心思想是把要操作的数列 \(a_i\) 分成若干长度相等的"块":修改/查询时对于整一块都在指定区间 \([L,R]\) 内的块整体修改/查询,对于只有块的一部分在指定区间内的暴力修改/查询. 由于不需要操作/查询具有 区间加法 等性质,分块比线段树.树状数组.ST表等数据结构具有更加灵活的应用. 先来看一道例题 数列分块入门 4,简而言之,就是要求实现区间加法&区间查询:线段树可以很轻松地实现这两个操作,

数列分块入门1-9 LibreOJ

数列分块入门1-9 LibreOJ 我也不知道为什么一个大二的ACM选手没学分块. 我怎么记得大一的时候,学长教给我的分块就只有 block 和 num 两个变量来着...好吧,应该是我没认真学.正好前两天朋友给学弟开课,乘机去蹭了一节课.然后...我还是不会哇,菜的一逼塌糊涂. 还是卿学姐好哇,多听几遍,睡得贼香. 分块原理 分块嘛,其实就是优雅的暴力,和莫队(不会)有点异曲同工的赶脚.通过将数组分成小块以降低复杂度. 通常情况下: 每个块的大小(block)为 \(\sqrt{n}\) 块数

loj 6278 6279 数列分块入门 2 3

参考:「分块」数列分块入门1 – 9 by hzwer 2 Description 给出一个长为\(n\)的数列,以及\(n\)个操作,操作涉及区间加法,询问区间内小于某个值\(x\)的元素个数. 思路 每个块内保持升序排列. 则块外暴力统计,块内二分查找分界点. 一些注意点,如: 要记录下标: 块外暴力修改完之后需要再排序: 在块内二分查找的值是\(c-tag[i]\)而非\(c\). Code #include <bits/stdc++.h> #define maxn 50010 #def

loj 6277 6280 数列分块入门 1 4

参考:「分块」数列分块入门1 – 9 by hzwer 1 Description 给出一个长为\(n\)的数列,以及\(n\)个操作,操作涉及区间加法,单点查值. 思路 用\(tag\)记录每个块整体的增量. Code #include <bits/stdc++.h> #define maxn 50010 #define F(i, a, b) for (int i = (a); i < (b); ++i) #define F2(i, a, b) for (int i = (a); i

LOJ6277~6285 数列分块入门

Portals 分块需注意的问题 数组大小应为\(N+\sqrt N\),因为最后一个块可能会超出\(N\)的范围.改成记录\(blk,fr,to\)就不用担心这个了 当操作的区间在一个块内时,要特判成暴力修改. 要清楚什么时候应该+tag[t] 最后一个块是越界的,注意是否有影响 数列分块入门 1 给出一个长为\(n\)的数列,以及\(n\)个操作,操作涉及区间加法,单点查值. //数列分块入门 1 #include <cstdio> #include <cmath> inlin

数列分块入门5 解题报告

占坑QAQ 数列分块系列目录 数列分块入门1 数列分块入门2 数列分块入门3 数列分块入门4 数列分块入门5 <- 数列分块入门6 数列分块入门7 数列分块入门8 数列分块入门9 蒲公英 公主的朋友 原文地址:https://www.cnblogs.com/louhancheng/p/10051160.html