zoukankan      html  css  js  c++  java
  • nyoj119 士兵杀敌(三)

     

    士兵杀敌(三)

    时间限制:2000 ms  |  内存限制:65535 KB
    难度:5
     
    描述

    南将军统率着N个士兵,士兵分别编号为1~N,南将军经常爱拿某一段编号内杀敌数最高的人与杀敌数最低的人进行比较,计算出两个人的杀敌数差值,用这种方法一方面能鼓舞杀敌数高的人,另一方面也算是批评杀敌数低的人,起到了很好的效果。

    所以,南将军经常问军师小工第i号士兵到第j号士兵中,杀敌数最高的人与杀敌数最低的人之间军功差值是多少。

    现在,请你写一个程序,帮小工回答南将军每次的询问吧。

    注意,南将军可能询问很多次。

     
    输入
    只有一组测试数据
    第一行是两个整数N,Q,其中N表示士兵的总数。Q表示南将军询问的次数。(1<N<=100000,1<Q<=1000000)
    随后的一行有N个整数Vi(0<=Vi<100000000),分别表示每个人的杀敌数。
    再之后的Q行,每行有两个正正数m,n,表示南将军询问的是第m号士兵到第n号士兵。
    输出
    对于每次询问,输出第m号士兵到第n号士兵之间所有士兵杀敌数的最大值与最小值的差。
    样例输入
    5 2
    1 2 6 9 3
    1 2
    2 4
    样例输出
    1
    7
    讲解:解决本题有两种方法,可以用线段树,也可以用RMQ算法,RMQ比较省时间,相对较优秀,
    其实线段树也是比较常规的方法;
    代码如下:线段树
     1 #include<iostream>
     2 #include<cstdio>
     3 #include<cstring>
     4 #include<algorithm>
     5 using namespace std;
     6 const int maxx=100010;
     7 int a[maxx];
     8 int big,small;
     9 struct node
    10 {
    11     int left,right;
    12     int Max,Min;
    13 }tree[3*maxx];
    14 void build(int l,int r,int v)
    15 {
    16     tree[v].left=l;
    17     tree[v].right=r;
    18     if(l==r)
    19     {
    20         tree[v].Max=a[l];
    21         tree[v].Min=a[l];
    22         return ;
    23     }
    24     int mid=(l+r)>>1;
    25     build(l,mid,v*2);
    26     build(mid+1,r,v*2+1);
    27     tree[v].Max=max(tree[2*v].Max,tree[2*v+1].Max);
    28     tree[v].Min=min(tree[2*v].Min,tree[2*v+1].Min);
    29 }
    30 void query(int a,int b,int v)
    31 {
    32     if(tree[v].left==a && tree[v].right==b)
    33     {
    34         if(tree[v].Max>big) big=tree[v].Max;
    35         if(tree[v].Min<small) small=tree[v].Min;
    36         return ;
    37     }
    38     int mid=(tree[v].left+tree[v].right)>>1;
    39     if(b<=mid)  query(a,b,2*v);
    40     else if(a>mid)  query(a,b,2*v+1);
    41     else
    42     {
    43      query(a,mid,v*2);
    44      query(mid+1,b,v*2+1);
    45     }
    46 }
    47 int main()
    48 {
    49     int n,m,q,l,r,i;
    50     scanf("%d%d",&n,&q);
    51         for(i=1;i<=n;i++)
    52            scanf("%d",&a[i]);
    53             build(1,n,1);
    54           for(i=0;i<q;i++)
    55           {
    56             scanf("%d%d",&l,&r);
    57             big=-1,small=100000005;
    58             query(l,r,1);
    59             printf("%d
    ",big-small);
    60           }
    61     return 0;
    62 }
    View Code

    RMQ算法:

     1 #include <iostream>
     2 #include <cmath>
     3 #include <algorithm>
     4 #include <cstdio>
     5 using namespace std;
     6 const int N=100000;
     7 
     8 int stmax[N][20],stmin[N][20],d[20];
     9 
    10 void init(int n)
    11 {
    12     int i,j;
    13     d[0]=1;
    14     for(i=1;i<=20;i++)d[i]=d[i-1]<<1;
    15     for(i=1;i<=n;i++){
    16         scanf("%d",&stmax[i][0]);
    17         stmin[i][0]=stmax[i][0];
    18     }
    19     int len = int(log(n*1.0)/log(2.0));
    20     for(j=1;j<=len;j++)
    21       for(i=1;i<=n;i++)
    22       if(i+d[j]-1<=n)
    23       {
    24             stmax[i][j]=max(stmax[i][j-1],stmax[i+d[j-1]][j-1]);
    25             stmin[i][j]=min(stmin[i][j-1],stmin[i+d[j-1]][j-1]);
    26       }
    27 }
    28 void rmq(int q) //
    29 {
    30     int x,y,len;
    31     while(q--)
    32     {
    33         scanf("%d%d",&x,&y);
    34         len=int(log(y-x+1.0)/log(2.0));
    35         int mmax=max(stmax[x][len],stmax[y-d[len]+1][len]);
    36         int mmin=min(stmin[x][len],stmin[y-d[len]+1][len]);
    37         printf("%d
    ",mmax-mmin);
    38     }
    39 }
    40 int main(){
    41       int n, q;
    42       scanf("%d%d",&n,&q);
    43       init(n);
    44       rmq(q);
    45     return 0;
    46 }
    View Code
  • 相关阅读:
    CVTE前端一面
    转载几篇看过的几篇使用技术博文
    Vue自定义指令
    node项目初始化的一些配置
    vue+element 实现在表格内插入其他组件,每行数据独立存储
    jquery如何实现当页面下拉到一定位置时,右下角出现回到顶部图标
    css实现垂直居中的方法整理
    css3兼容性检测工具
    css变换与动画详解
    手机访问电脑端网站偏问题解决
  • 原文地址:https://www.cnblogs.com/lovychen/p/3436423.html
Copyright © 2011-2022 走看看