题目大意

给出一个长度为 \(N\) 的序列 \(a\) 需要构造出一个长度为 \(N\) 的序列 \(h\) 使得 \(\forall i \in \{1,2,\ldots ,N\} h_i \leq a_i\) 且 \(\forall i \in \{2,3,\ldots ,N-1\}\),\(\nexists \forall j \in \{1,2,\ldots ,i-1\},k \in \{i+1,i+2 \ldots N\},h_j > h_i < h_k\),求出并输出 \(\sum_{i=1}^{N}h_i\) 达到最大值时的 \(h\) 序列.

再简化一下题意,就是需要找出一个序列 \(h\),\(h_i\leq a_i\),且以 \(h\) 中的某一个元素看来,前面和后面的元素都是单调的.

本题在数据范围上分为两个部分,所以就写一下两种不同的做法.

Part 1

分析

\(N\) 的范围不大,只有 \(1000\) 于是很容易想到时间复杂度应该是 \(O(N^2)\) 的,可以想到先枚举最高的位置的点,在向两边计算,可以很容易想到最高位置的 \(h_i=a_i\) 而且当最高位置确定以后,为了使得和最大,所以整一个序列也是确定的,那么就可以想出最开始的 \(O(N^2)\) 的做法了

代码

#include<bits/stdc++.h>
#define REP(i,first,last) for(int i=first;i<=last;++i)
#define DOW(i,first,last) for(int i=first;i>=last;--i)
using namespace std;
int N,h[114514];
int top;
long long ans[114514];
int main()
{
scanf("%d",&N);
REP(i,1,N)
{
scanf("%d",&h[i]);
}
int now;
long long answer=0;
long long sum;
REP(i,1,N)//枚举最高的位置
{
now=h[i];//记录当前的位置
sum=h[i];
DOW(j,i-1,1)//向前扫
{
sum+=min(h[j],now);
now=min(h[j],now);//这里的now因为要保证单调不下降,所以需要一直取min
}
now=h[i];//向后扫同理
REP(j,i+1,N)
{
sum+=min(h[j],now);
now=min(h[j],now);
}
if(sum>answer)//记录下最大值,以及最大值时时哪一个位置达到最大值了
{
answer=sum;
top=i;
}
}
//最后只需要将达到最大值时的序列输出就好了
ans[top]=h[top];
now=h[top];
DOW(i,top-1,1)
{
ans[i]=min(h[i],now);
now=min(h[i],now);
}
now=h[top];
REP(i,top+1,N)
{
ans[i]=min(h[i],now);
now=min(h[i],now);
}
REP(i,1,N)
{
printf("%d ",ans[i]);
}
return 0;
}

Part 2

分析

\(N\) 的范围变大了很多达到了 \(500000\) 为了防止题解清一色的都是单调栈,这里是一种不是很常见的做法.

可以发现在第一种方法中如果 \(j \leq i,h_j \leq h_i\) 时 \(1\) 到 \(j\) 的最大值肯定是被计算过的,于是就可以想到用一个数组 \(L\) 维护从 \(1\) 开始带第 \(i\) 个位置时最高处为 \(i\) 切 \(h_1\) 到 \(h_i\) 为单调不下降时 \(h_1\) 到 \(h_i\) 和的最大值,计算这个数组需要分两种情况.

  1. 第一种情况时 \(a_{i-1} \leq a_i\),这样 \(L_i=L_{i-1}+a_i\).
  2. 第二种情况就比较特殊了,没法直接从 \(L_{i-1}\) 计算出来,需要找到上一个小于等于 \(a_i\) 的位置 \(j\),那么 \(L_i=(j-i)*a_i+L_j\),即在这之间的大于 \(a_i\) 的位置的高度都是 \(a_i\) 这样就可以保证是单调不下降.

于是就要想办法找到上一个小于 \(a_i\) 的位置,显然不可以用 \(O(N)\) 的做法,那么就可以用一颗线段树维护最小值,在线段树上二分找到上一个位置.

用同样的方法计算出一个数组 \(R\)(即从 \(h_i\) 开始到 \(h_N\) 中单调不上升的和的最大值)与 \(L\) 同理,这样以第 \(i\) 个位置为最高值时总和就变成了 \(L_i+R_i-a_i\) 取出最大值以后只要像方法一一样输出就好了.

代码

#include<bits/stdc++.h>
#define REP(i,first,last) for(int i=first;i<=last;++i)
#define DOW(i,first,last) for(int i=first;i>=last;--i)
using namespace std;
const int MAXN=6e5+7;
int N,M;
int arr[MAXN];
long long L[MAXN],R[MAXN];
int ans[MAXN];
//线段树部分,不多讲
struct SegmentTree
{
int min;
}sgt[MAXN*4];
#define LSON (now<<1)
#define RSON (now<<1|1)
#define MIDDLE ((left+right)>>1)
#define LEFT LSON,left,MIDDLE
#define RIGHT RSON,MIDDLE+1,right
void PushUp(int now)
{
sgt[now].min=min(sgt[LSON].min,sgt[RSON].min);
}
void Build(int now=1,int left=1,int right=N)
{
if(left==right)
{
sgt[now].min=arr[left];
return;
}
Build(LEFT);
Build(RIGHT);
PushUp(now);
}
int QueryL(int l,int r,int num,int now=1,int left=1,int right=N)//查询下一个小于num的值的位置
{
if(r<left||right<l)return -1;
if(sgt[now].min>num)
{
return -1;
}
if(left==right)
{
return left;
}
int result=QueryL(l,r,num,LEFT);//因为是下一个,所以优先查询左边
if(result!=-1)
{
return result;
}
return QueryL(l,r,num,RIGHT);
}
int QueryR(int l,int r,int num,int now=1,int left=1,int right=N)//查询上一个小于num的值的位置
{
if(r<left||right<l)return -1;
if(sgt[now].min>num)
{
return -1;
}
if(left==right)
{
return left;
}
int result=QueryR(l,r,num,RIGHT);
if(result!=-1)
{
return result;
}
return QueryR(l,r,num,LEFT);
}
#undef LSON
#undef RSON
#undef MIDDLE
#undef LEFT
#undef RIGHT
int main()
{
scanf("%d",&N);
REP(i,1,N)
{
scanf("%d",&arr[i]);
}
Build();
REP(i,1,N)
{
if(arr[i]>=arr[i-1])//计算L时的第一种情况
{
L[i]=L[i-1]+1ll*arr[i];
}
else
{
int top=QueryR(1,i-1,arr[i]);//找到上一个小于的位置
if(top==-1)L[i]=1ll*i*arr[i];//不存在就直接计算
else
L[i]=1ll*(i-top)*arr[i]+L[top];//存在按第二种情况计算
}
}
DOW(i,N,1)//计算R同理
{
if(arr[i]>=arr[i+1])
{
R[i]=R[i+1]+1ll*arr[i];
}
else
{
int top=QueryL(i+1,N,arr[i]);
if(top==-1)R[i]=1ll*(N-i+1)*arr[i];
else
R[i]=1ll*(top-i)*arr[i]+R[top];
}
}
int top=1;
long long answer=0;
long long sum;
REP(i,1,N)
{
sum=L[i]+R[i]-arr[i];
if(sum>answer)//找到最大位置
{
answer=sum;
top=i;
}
}
//同样方法输出
ans[top]=arr[top];
int now=arr[top];
DOW(i,top-1,1)
{
ans[i]=min(arr[i],now);
now=min(arr[i],now);
}
now=arr[top];
REP(i,top+1,N)
{
ans[i]=min(arr[i],now);
now=min(arr[i],now);
}
REP(i,1,N)
{
printf("%d ",ans[i]);
}
return 0;
}

「CF1313C Skyscrapers」的更多相关文章

  1. 前端构建工具之gulp(一)「图片压缩」

    前端构建工具之gulp(一)「图片压缩」 已经很久没有写过博客了,现下终于事情少了,开始写博吧 今天网站要做一些优化:图片压缩,资源合并等 以前一直使用百度的FIS工具,但是FIS还没有提供图片压缩的 ...

  2. fir.im Weekly - 如何打造 Github 「爆款」开源项目

    最近 Android 转用 Swift 的传闻甚嚣尘上,Swift 的 Github 主页上已经有了一次 merge>>「Port to Android」,让我们对 Swift 的想象又多 ...

  3. 更新日志 - fir.im「高级统计」功能上线

    距离 2016 年到来只剩 10 个日夜,fir.im 也准备了一些新鲜的东西,比如「高级统计」功能和「跳转应用商店」功能,帮助你更好地管理.优化应用,欢迎大家试用反馈:) 新增高级统计功能 这次更新 ...

  4. Notepad++ 开启「切分窗口」同时检视、比对两份文件

    Notepad++ 是个相当好用的免费纯文本编辑器,除了内建的功能相当多之外,也支持外挂模块的方式扩充各方面的应用.以前我都用 UltraEdit 跟 Emeditor,后来都改用免费的 Notepa ...

  5. 「zigbee - 1」工欲善其事必先利其器 - IAR for 8051 IDE customization

    最近在实验室做一些 Zigbee 相关的事情,然而一直没在博客上记录啥东西,也不像原来在公司有动力在 Confluence wiki 上扯东扯西.直到前些阵子,跑到 feibit 论坛上(国内较大的一 ...

  6. 「C语言」文件的概念与简单数据流的读写函数

    写完「C语言」单链表/双向链表的建立/遍历/插入/删除 后,如何将内存中的链表信息及时的保存到文件中,又能够及时的从文件中读取出来进行处理,便需要用到”文件“的相关知识点进行文件的输入.输出. 其实, ...

  7. 「C语言」Windows+EclipseCDT下的C语言开发环境准备

    之前写过一篇 「C语言」在Windows平台搭建C语言开发环境的多种方式 ,讨论了如何在Windows下用DEV C++.EclipseCDT.VisualStudio.Sublime Test.Cl ...

  8. 如何对抗 WhatsApp「蓝色双勾」-- 3 个方法让你偷偷看讯息

    WhatsApp 强制推出新功能「蓝色双勾 (✔✔)」 ,让对方知道你已经看过讯息.一众用户反应极大,因为以后不能再藉口说未看到讯息而不回覆.究竟以后 WhatsApp 是否真的「更难用」? 幸好还有 ...

  9. FileUpload控件「批次上传 / 多档案同时上传」的范例--以「流水号」产生「变量名称」

    原文出處  http://www.dotblogs.com.tw/mis2000lab/archive/2013/08/19/multiple_fileupload_asp_net_20130819. ...

随机推荐

  1. maven一直加载2.0.0.M7 的 config server 失败

    之前学习的时候使用F版的SpringBoot管理项目依赖一直好好的,今天不知idea为何抽疯,一直加载失败,各种重启,清除,没用 只能像之前学习注册consul 时将F版的SpringBoot 改为G ...

  2. docker为镜像添加SSH服务

    启动并进入容器中 这里用db1容器完成实验. 安装openssh服务和修改sshd配置文件 安装openssh yum install openssh-server openssh-clients - ...

  3. Redis07——Redis五大数据类型 set

    set Redis中set可以自动排重(不会出现重复数据),提供了判断某个成员是否在一个set集合内的重要接口(list没有此功能) 无序集合,底层是一个value为null的hash表,添加.删除. ...

  4. bugku 社工进阶

    首先看到的是 由于之前知道有bugku的百度吧   并且这个是一个社工题所以可以试一下这个百度吧 进入百度吧然后会见到 这句话的意思是要我们登录这个账号 但是我们只有账号没有密码  如果爆破的话很有可 ...

  5. 概率dp poj 3071

    题目首先给出一个n,表示比赛一共进行n轮,那么队伍就有2^n只队伍输入一个2^n*2^n的矩阵,p[i][j]代表队伍i打败队伍j的概率dp[i][j]代表第i轮比赛的时候,队伍j赢的概率首先初始化时 ...

  6. CSS--box

    width is content width height is content height set margin and padding zero leads box to the same wi ...

  7. shell笔记1

  8. A. DZY Loves Chessboard

    DZY loves chessboard, and he enjoys playing with it. He has a chessboard of n rows and m columns. So ...

  9. 你知道for(;;) vs. while(true)那个更快吗?

    来来来, for(;;) vs. while(true) 有什么区别?从java的语义上来说,他们是一模一样的.为何怎么说? 开始我们先测试for(;;) package com.tony.test; ...

  10. Spring IoC 源码分析 (基于注解) 之 包扫描

    在上篇文章Spring IoC 源码分析 (基于注解) 一我们分析到,我们通过AnnotationConfigApplicationContext类传入一个包路径启动Spring之后,会首先初始化包扫 ...