TYVJ P1039 【忠诚2】

题目描述

老管家是一个聪明能干的人。他为财主工作了整整10年,财主为了让自已账目更加清楚。要求管家每天记k次账,由于管家聪明能干,因而管家总是让财主十分满意。但是由于一些人的挑拨,财主还是对管家产生了怀疑。于是他决定用一种特别的方法来判断管家的忠诚,他把每次的账目按1,2,3…编号,然后不定时的问管家问题,问题是这样的:在a到b号账中最少的一笔是多少?为了让管家没时间作假他总是一次问多个问题。

在询问过程中账本的内容可能会被修改。

输入

输入中第一行有两个数m,n表示有m(m< =100000)笔账,n表示有n个问题,n<=100000。 接下来每行为3个数字,第一个p为数字1或数字2,第二个数为x,第三个数为y。当p=1,则查询[x,y]区间;当p=2,则改变第x个数为y。

输出

输出文件中为每个问题的答案。具体查看样例。

样例输入

10 3
1 2 3 4 5 6 7 8 9 10
1 2 7
2 2 0
1 1 10

样例输出

2 0

这题显然可以用线段树解决,但是作为一道这么简单的题目,我们尝试一下其它的做法。

最近看了一下白书写的分桶法,就拿这题来试试效果怎样。

分桶法的时间复杂度一般是带根号的。

在这里,我用√n个桶,则每个桶里都有n/√n=√n个数值。

对于每次操作,时间复杂度为 O(√n)(至于为什么下面注释里会说)。

所以总的时间复杂度为 O(m*√n)。

 1 #include <cstdio>
 2 #include <cstring>
 3 
 4 //maxsqr既是桶的数量,也是每个桶里存的元素的数量 
 5 const int maxn=100000, maxsqr=400;
 6 
 7 int n,m;
 8 //a数组为该数的值  
 9 int a[maxn];
10 //bucket[i]表示第i个桶里的最小值
11 //即为a[i*maxsqr]~a[(i+1)*maxsqr-1]这个区间的最小值  
12 int bucket[maxsqr];
13 
14 int max(int x, int y) { return x>y?x:y; }
15 
16 int min(int x, int y) { return x<y?x:y; }
17 
18 void init() {
19     memset(bucket,0x7F,sizeof(bucket)); //对每个桶进行初始化  
20     
21     scanf("%d%d",&n,&m);
22     for (int i=0; i<n; i++) {
23         scanf("%d",a+i);
24         //i号元素所在的桶的编号为i/maxsqr 
25         bucket[i/maxsqr]=min(bucket[i/maxsqr],a[i]);
26     }
27 }
28 
29 //更新操作 
30 void update(int x, int y) {
31     a[x]=y; //先单独更新数值  
32     int t=x/maxsqr; //该数值所在的桶的编号 
33     
34     //把该数值所在的桶表示的区间再扫过一遍,重置最小值 
35     //因为桶内不超过sqrt(n)个元素 
36     //所以更新操作的时间复杂度为 O(sqrt(n)) 
37     bucket[t]=0x7F7F7F7F;
38     //特别注意,此处i<n是为了保证不访问到没有数值的区域 
39     //如果访问到,一来会RE,二来那里的a[i]=0,桶内的最小值会错 
40     for (int i=t*maxsqr; i<(t+1)*maxsqr&&i<n; i++)
41         bucket[t]=min(bucket[t],a[i]);
42 }
43 
44 //查询操作 
45 int query(int x, int y) {
46     int res=0x7F7F7F7F;
47     //左端所在的桶:t1, 右端所在的桶:t2 
48     int t1=x/maxsqr, t2=y/maxsqr;
49     
50     //没有被完全覆盖在桶里的,也就是左右端所在的桶
51     //一个个扫过去,暴力求解最小值 
52     //因为多出来的元素最多2*sqrt(n)个 
53     //所以时间复杂度控制在 O(sqrt(n)) 
54     for (int i=x; i<(t1+1)*maxsqr&&i<=y; i++)
55         res=min(res,a[i]);
56     for (int i=max(x,t2*maxsqr); i<=y; i++)
57         res=min(res,a[i]);
58     
59     //查询的区间完全覆盖了这些桶所保存的区间最小值 
60     //则只要与已经存好的桶的最小值作比较就好了  
61     //因为桶只有sqrt(n)个 
62     //所以时间复杂度亦为 O(sqrt(n))  
63     //这样,整个查询操作的复杂度就为 O(sqrt(n)) 
64     for (int i=t1+1; i<t2; i++)
65         res=min(res,bucket[i]);
66     return res;
67 }
68 
69 int main() {
70     init();
71     for (int i=1; i<=m; i++) {
72         int p,x,y;
73         scanf("%d%d%d",&p,&x,&y);
74         //因为数组是从0开始用的,所以表示下标的数都应该-1 
75         if (p==2)
76             update(x-1,y);
77         else printf("%d ",query(x-1,y-1));
78     }
79     return 0;
80 }
原文地址:https://www.cnblogs.com/tweetuzki/p/8320992.html