zoukankan      html  css  js  c++  java
  • 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 }
  • 相关阅读:
    UIButton(在代码中使用)
    UIButton的常见设置
    设置按钮的背景图片
    Java Web学习总结(15)——JSP指令
    Java Web学习总结(14)——JSP基础语法
    Java Web学习总结(14)——JSP基础语法
    Oracle学习总结(9)—— Oracle 常用的基本操作
    Oracle学习总结(9)—— Oracle 常用的基本操作
    Oracle学习总结(10)——45 个非常有用的 Oracle 查询语句
    Oracle学习总结(10)——45 个非常有用的 Oracle 查询语句
  • 原文地址:https://www.cnblogs.com/tweetuzki/p/8320992.html
Copyright © 2011-2022 走看看