2017 10 01国庆节大礼包 四校联考

1.积木大赛

(block.pas/c/cpp)

【问题描述】

为了庆祝国庆,厦门一中举办了一年一度的“积木大赛”。

在2013年NOIP大赛中,夏夏同学己经搭建了宽度为n的大厦,其中第i块高度为hi。今年比赛的内容是对其NOIP2013搭建大厦进行扩建,使用的材料也都是体积为1正方体积木。

今年搭建的规则是:如果要在某一个位置上放一个积木,必须满足它的左下、下方、右下都有积木(用二维坐标a表示,如果要在a[i,j]位置放积木,那么a[i-1,j-1]、a[i,j-1]、a[i+1,j-1]必须要有积木)。

 

  如果搭的积木大厦越高,夏夏同学就会觉得越有成就感,现有m个积木,问你能搭建的最大高度是多少?

【输入】

第一行两个用空格隔开的整数n和m,分别表示己搭好的宽度和可以使用的积木数量。

后面有n行,每行一个整数hi表示己搭建的第i列积木的高度。

【输出】

一个整数,表示能搭建的最大高度。

【输入输出样例】

样例1

样例2

block.in

block.out

block.in

block.out

8 4

3

4

2

1

3

3

2

4

5

3 100

3

3

3

4

【数据说明】

    30%的数据满足:n<=10;m<=1000。

    50%的数据满足:n<=100;m<=1000,000。

70%的数据满足:n<=1000;m<=10,000,000。

    80%的数据满足:n<=10,000;m<=100,000,000。

100%的数据满足:n<=100,000;m<=1000,000,000。

 

 

题解:

算法一:对于50%的数据:

①从高到低枚举每一个高度看是否能达到, 时间复杂度O(n);

②对于每一个高度,枚举最高点在哪一列,时间复杂度O(n);

③从当前枚举的最高列往两边递减判断是否合法,时间复杂度O(n);

总时间复杂度O(n3)。

 

算法二:对于70%的数据:

在算法一中的第二步求高度时,使用二分答案,时间复杂度降为O(logn)。总时间复杂度为O(n2logn)。

 

算法三:100%的数据:

  对于第二步和第三步,根据单调性,我们用L[i]表示在高度为H时第I列达到高度H时最左端的位置,R[i]为最右端的位置。通过O(n)的时间得到L,R,总的时间复杂度为O(nlogn)。

 

 1 #include <iostream>
 2 #include <algorithm>
 3 #include <cmath>
 4 #include <stdio.h>
 5 #include <string>
 6 #include <string.h>
 7 #include <numeric>
 8 using namespace std;
 9 int a[500001];
10 long long s[500001];
11 int l[500001];
12 int r[500001];int n,m;
13 int read()
14 {
15     char ch=getchar();
16     int t=0;
17     int f=1;
18     while(ch>'9' || ch<'0')
19       {ch=getchar();}
20     while(ch>='0' && ch<='9')
21       t=(t<<3)+(t<<1)+ch-'0',ch=getchar();
22     return t;
23 }
24 bool check(int h)
25 {
26     memset(l,0,sizeof l);memset(r,63,sizeof r);
27     int i;
28     int inf=r[0];
29     for (i=1;i<=n;i++)
30         if (i+h-a[i]<=n)
31             l[i+h-a[i]]=i;
32     for (i=n;i>=1;i--)
33         if (i+a[i]-h>0)
34           r[i+a[i]-h]=i;
35     
36     for (i=1;i<=n;i++)
37       l[i]=max(l[i],l[i-1]);
38     for (i=n;i>=1;i--)
39       r[i]=min(r[i],r[i+1]);
40     for (i=1;i<=n;i++)
41       {
42           if (!l[i] || inf==r[i]) continue;
43           if((1ll*(a[l[i]]+h)*(i-l[i]+1)>>1)+(1ll*(a[r[i]]+h-1)*(r[i]-i)>>1)-(s[r[i]]-s[l[i]-1]) <= m)
44             return 1;
45       }
46     return 0;
47 }
48 int main()
49 {
50     
51     n=read();
52     m=read();
53     int i;int l=0;
54     for (i=1;i<=n;i++)
55       a[i]=read(),s[i]=s[i-1]+a[i],l=max(l,a[i]);
56     int r=2000000000;
57     int mid;
58     while(l<r)
59       {
60           mid=l+((r-l)>>1);
61           if (check(mid))
62             l=mid+1;
63           else r=mid;
64       }
65     printf("%d\n",l-1);
66     return 0;
67 }
block

 

posted @ 2017-10-02 09:19  yz12138  阅读(226)  评论(1编辑  收藏  举报