两个分块题

题目一:给出一个长为 的数列,以及 n 个操作,操作涉及区间加法,询问区间内小于某个值 x 的元素个数。

 分析  :将数列分成sqrt(n)块,每块维护一个懒惰标记、一个有序表(vector实现)即可。

    操作的时候,先暴力处理左右两端不完整的块,再处理中间完整的块,注意懒惰标记的释放。

    区间加法O(√n),询问区间内小于某个值 x 的元素个数O(√n*log√n)

代码:

 1 #include<iostream>
 2 #include<cstdio>
 3 #include<cmath>
 4 #include<cstdlib>
 5 #include<algorithm>
 6 #include<cstring>
 7 #include<string>
 8 #include<vector>
 9 #include<map>
10 #include<set>
11 #include<queue>
12 #define N 100005
13 using namespace std;
14 typedef long long ll;
15 int a[N],b[N];
16 int mark[505];
17 vector<int>vt[505];
18 int n,blo;
19 void resort(int x)
20 {
21     vt[x].clear();
22     for(int i=blo*(x-1)+1;i<=min(n,blo*x);i++)
23         vt[x].push_back(a[i]);
24     sort(vt[x].begin(),vt[x].end());
25 }
26 int main()
27 {
28    
29    cin>>n;
30    blo=sqrt(n);
31     for(int i=1;i<=n;i++)
32        scanf("%d",&a[i]);
33     for(int i=1;i<=n;i++)
34     {
35         b[i]=(i-1)/blo+1;
36         vt[b[i]].push_back(a[i]);
37     }
38     for(int i=1;i<=b[n];i++)
39     {
40         sort(vt[i].begin(),vt[i].end());
41     }
42     for(int i=1;i<=n;i++)
43     {
44         int op,l,r,c;
45         scanf("%d%d%d%d",&op,&l,&r,&c);
46         if(op==0)
47         {
48             for(int i=l;i<=min(blo*b[l],r);i++)
49                 a[i]+=c;
50             resort(b[l]);
51             for(int i=b[l]+1;i<=b[r]-1;i++)
52             {
53                 mark[i]+=c;
54             }
55             if(b[l]!=b[r])
56             {
57                 for(int i=blo*(b[r]-1)+1;i<=r;i++)
58                     a[i]+=c;
59                 resort(b[r]);
60             }
61         }
62         else if(op==1)
63         {    int num=0;
64             ll cc=c*c;
65             for(int i=l;i<=min(blo*b[l],r);i++)
66                 if(cc>a[i]+mark[b[l]]) num++;
67     
68             for(int i=b[l]+1;i<=b[r]-1;i++)
69             {
70                 num+=lower_bound(vt[i].begin(),vt[i].end(),cc-mark[i])-vt[i].begin();
71             }
72     
73             if(b[l]!=b[r])
74             {
75                 for(int i=blo*(b[r]-1)+1;i<=r;i++)
76                     if(cc>a[i]+mark[b[r]]) num++;
77             }
78             printf("%d\n",num);
79         }
80     }     
81     return 0;
82 } 

 

题目二:给出一个长为 n 的数列,以及 n 个操作,操作涉及单点插入,单点询问,数据随机生成。

 分析 :将数列分成sqrt(n)块,每块维护一个链表。由于单点插入会使原来的编号发生改变,所以每块还需维护好该块的区间(l,r)。

    查找的时候先遍历找到属于哪个块,再在块中遍历即可,单次O(√n)。

    一直插入操作可能使某个块过长,这时需要将其切成两个块(重构)

代码:

 1 #include<iostream>
 2 #include<cstdio>
 3 #include<cmath>
 4 #include<cstdlib>
 5 #include<algorithm>
 6 #include<cstring>
 7 #include<string>
 8 #include<vector>
 9 #include<map>
10 #include<set>
11 #include<queue>
12 #include<list>
13 #define N 100005
14 using namespace std;
15 typedef long long ll;
16 int a[N];
17 
18 typedef struct NODE{
19     int l, r;
20     list<int>lt;
21 }node;
22 list<node>head(405);
23 int main()
24 {
25     int n,blo;
26     cin>>n;
27     blo=sqrt(n);
28     for(int i=1;i<=n;i++)
29         scanf("%d",&a[i]);
30     int m=(n-1)/blo+1;
31     list<node>::iterator p=head.begin();
32     for(int i=1;i<=m;i++)
33     {    
34         p->l=(i-1)*blo+1;
35         p->r=min(n,i*blo);
36         for(int k=p->l;k<=p->r;k++)
37         {
38             p->lt.push_back(a[k]);
39         }
40         p++;    
41     }
42     for(int i=1;i<=n;i++)
43     {
44         int op,l,r,c;
45         scanf("%d%d%d%d",&op,&l,&r,&c);
46         if(op==0)
47         {
48             p=head.begin();
49             while(p!=head.end())
50             {
51                 if(p->l<=l&&l<=p->r) break;
52                 p++;
53             }
54             int tot=0;
55             list<int>::iterator p2=p->lt.begin(); 
56             while(p2!=p->lt.end())
57             {
58                 if(p->l+tot==l) {p->lt.insert(p2,r);break;}
59                 p2++;
60                 tot++;
61             }
62             p->r+=1;
63             p++;
64             for(;p!=head.end();p++)
65             {
66                 p->l+=1,p->r+=1;
67             }
68         }
69         else if(op==1)
70         {
71             p=head.begin();
72             while(p!=head.end())
73             {
74                 if(p->l<=r&&r<=p->r) break;
75                 p++;
76             }
77             int tot=0;
78             list<int>::iterator p2=p->lt.begin(); 
79             while(p2!=p->lt.end())
80             {
81                 if(p->l+tot==r) {printf("%d\n",*p2);break;}
82                 p2++;
83                 tot++;
84             }
85         }
86         
87         
88     }
89     
90     return 0;
91 }

 

    

posted @ 2018-07-10 09:10  hzhuan  阅读(224)  评论(0编辑  收藏  举报