Description

有\(N(N\leq 10^5)\)个排列在一条线上的城市,每个城市有\(val_i\)个景点。每天你可以选择在当前城市\(i\)游览景点,或者前往城市\(i-1\)或城市\(i+1\)。给定起点和天数,请最大化游览的景点。一个城市的景点最多只会被游览一次。

Solution

1A IOI真是劲啊

因为这个游览城市没有“过期”一说,所以我们没必要蛇形走位,也就是说,我们行进的路线只有四种情况:一直向右,一直向左,先向左再向右,先向右再向左。这样可以求出来四个数组。

以求一直向右为例,设\(f[i]\)表示一直向右走\(i\)天的最大收益。观察到决策点是单调的,我们可以分治\(solve(l,r,x,y)\)表示要求\(f[l...r]\),决策点在\([x,y]\)。具体求法可以用主席树查前\(K\)大的值来实现。

其他三个求法也是类似的。还要注意一点就是我们强制让后两种情况的\(f[i]\)表示经过\(i\)天又回到起点的最大收益,然后用回到起点的最大收益加上从起点出发的最大收益更新答案就行了。

Code

#include<set>
#include<map>
#include<cmath>
#include<queue>
#include<cctype>
#include<vector>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
using std::min;
using std::max;
using std::swap;
using std::vector;
const int N=1e5+5;
const int M=N*3;
typedef double db;
typedef long long ll;
#define pb(A) push_back(A)
#define pii std::pair<int,int>
#define mp(A,B) std::make_pair(A,B)
#define ls ch[cur][0]
#define rs ch[cur][1] ll a[5][M];
int n,s,m,len,tot,val[N],g[N];
int root[N],ch[N*20][2];ll sum[N*20][2]; int getint(){
int X=0,w=0;char ch=0;
while(!isdigit(ch))w|=ch=='-',ch=getchar();
while( isdigit(ch))X=X*10+ch-48,ch=getchar();
if(w) return -X;return X;
} int modify(int pre,int l,int r,int c){
int cur=++tot;
sum[cur][1]=sum[pre][1]+g[c];sum[cur][0]=sum[pre][0]+1;ls=ch[pre][0];rs=ch[pre][1];
if(l==r) return cur;
int mid=l+r>>1;
if(c<=mid) ls=modify(ch[pre][0],l,mid,c);
else rs=modify(ch[pre][1],mid+1,r,c);
return cur;
} ll query(int pre,int cur,int l,int r,int k){
if(sum[cur][0]-sum[pre][0]<=k) return sum[cur][1]-sum[pre][1];
if(l==r) return (ll)g[l]*k;
int mid=l+r>>1;
if(sum[rs][0]-sum[ch[pre][1]][0]>k) return query(ch[pre][1],ch[cur][1],mid+1,r,k);
return sum[rs][1]-sum[ch[pre][1]][1]+query(ch[pre][0],ls,l,mid,k-sum[rs][0]+sum[ch[pre][1]][0]);
} int abs(int x){
return x<0?-x:x;
} void solve(int l,int r,int x,int y,int pd){
if(l>r) return;
int mid=l+r>>1,idx=0;//一共走mid天
for(int i=x;i<=y;i++){
ll t;
if(pd==1) t=query(root[s-1],root[i],1,len,mid-abs(i-s));
else if(pd==2) t=query(root[i-1],root[s-1],1,len,mid-abs(i-s));
else if(pd==3) t=query(root[s-1],root[i],1,len,mid-abs(i-s)*2);
else t=query(root[i-1],root[s-1],1,len,mid-abs(i-s)*2);
if(t>a[pd][mid]) a[pd][mid]=t,idx=i;
}
if(pd==1 or pd==3) solve(l,mid-1,x,idx,pd),solve(mid+1,r,idx,y,pd);
else solve(l,mid-1,idx,y,pd),solve(mid+1,r,x,idx,pd);
} signed main(){
n=getint(),s=getint()+1,m=getint();
for(int i=1;i<=n;i++)g[i]=val[i]=getint();
std::sort(g+1,g+1+n);len=std::unique(g+1,g+1+n)-g-1;
for(int i=1;i<=n;i++){
val[i]=std::lower_bound(g+1,g+1+len,val[i])-g;
root[i]=modify(root[i-1],1,len,val[i]);
}
solve(1,m,s,n,1);solve(1,m,1,s,2);
solve(1,m,s,n,3);solve(1,m,1,s,4);
for(int i=1;i<=m;i++)
for(int j=1;j<=4;j++)
a[j][i]=max(a[j][i],a[j][i-1]);
ll ans=max(a[1][m],a[2][m]);
for(int i=1;i<m;i++)
ans=max(ans,max(a[3][i]+a[2][m-i],a[4][i]+a[1][m-i]));
printf("%lld\n",ans);
return 0;
}

[IOI2014] 假期的更多相关文章

  1. 【BZOJ4367】[IOI2014]holiday假期 分治+主席树

    [BZOJ4367][IOI2014]holiday假期 Description 健佳正在制定下个假期去台湾的游玩计划.在这个假期,健佳将会在城市之间奔波,并且参观这些城市的景点.在台湾共有n个城市, ...

  2. [IOI2014]holiday假期(分治+主席树)

    题目描述 健佳正在制定下个假期去台湾的游玩计划.在这个假期,健佳将会在城市之间奔波,并且参观这些城市的景点.在台湾共有n个城市,它们全部位于一条高速公路上.这些城市连续地编号为0到n-1.对于城市i( ...

  3. BZOJ4367 : [IOI2014]holiday假期

    设 $fl[i]$表示从$S$向左走,用了不超过$i$天且不回头的最大收益. $fr[i]$表示从$S$向右走,用了不超过$i$天且不回头的最大收益. $gl[i]$表示从$S$向左走,用了不超过$i ...

  4. luogu P5892 [IOI2014]holiday 假期 决策单调性优化dp 主席树

    LINK:holiday 考虑第一个subtask. 容易想到n^2暴力枚举之后再暴力计算答案. 第二个subtask 暴力枚举终点可以利用主席树快速统计答案. 第三个subtask 暴力枚举两端利用 ...

  5. [BZOJ4367][IOI2014]Holiday(决策单调性+分治+主席树)

    4367: [IOI2014]holiday假期 Time Limit: 20 Sec  Memory Limit: 64 MBSubmit: 421  Solved: 128[Submit][Sta ...

  6. BZOJ1433 ZJOI2009 假期的宿舍 二分图匹配

    1433: [ZJOI2009]假期的宿舍 Time Limit: 10 Sec  Memory Limit: 162 MBSubmit: 2375  Solved: 1005[Submit][Sta ...

  7. 假期实践作业:从IT角度看地铁

    实习时间:2016/02/23——2016/02/26 实习地点:京港地铁14号线 实习报告: 大学四年过得真快,转眼就大三了,大学前两年半的生活可谓多姿多彩,从不懂计算机到对编程感兴趣,期待得最多的 ...

  8. 2055 [ZJOI2009]假期的宿舍

    P2055 [ZJOI2009]假期的宿舍 题目描述 学校放假了 · · · · · · 有些同学回家了,而有些同学则有以前的好朋友来探访,那么住宿就是一个问题.比如 A 和 B 都是学校的学生,A ...

  9. 假期(codevs 3622)

    题目描述 Description 经过几个月辛勤的工作,FJ决定让奶牛放假.假期可以在1-N天内任意选择一段(需要连续),每一天都有一个享受指数W.但是奶牛的要求非常苛刻,假期不能短于P天,否则奶牛不 ...

随机推荐

  1. Linux配置ntp时间服务器(全)

    时间服务器作用: 大数据产生与处理系统是各种计算设备集群的,计算设备将统一.同步的标准时间用于记录各种事件发生时序, 如E-MAIL信息.文件创建和访问时间.数据库处理时间等. 大数据系统内不同计算设 ...

  2. ActiveMQ_3Java实现

    Java实现 添加相应的jar包 <dependency> <groupId>org.apache.activemq</groupId> <artifactI ...

  3. 日期Data类,日历类Calendar

    用于得到当前时间,和设置日期类数据 public void testDate() { // 创建一个日期对象 Date date = new Date(); /** * 从1900年1月1日 00:0 ...

  4. Linux 根据PID找到相应应用程序的运行目录

    1.找到运行程序的PID # ps aux | grep redis root pts/ S+ : : grep redis root ? Ssl Aug30 : redis-server *: # ...

  5. Codeforces Round #512 (Div. 2) D. Vasya and Triangle

    参考了别人的思路:https://blog.csdn.net/qq_41608020/article/details/82827632 http://www.cnblogs.com/qywhy/p/9 ...

  6. ReactNative学习笔记(六)集成视频播放

    概述 视频播放可以自己写原生代码实现,然后注入JS.如果对视频播放没有特殊要求的话,可以直接使用现成插件. 到官方推荐的插件网站搜索找到下载量第一的插件:react-native-video. 安装 ...

  7. cad2008 参照面板 在位编辑后无法使用

    有时候在位编辑之后,保存在位编辑了,可是参照面板无法动了,这个时候只要保存一下,然后ctrl+z,就可以解决了..蛮神奇的,我还没想到如何用代码去搞定它..

  8. Java 实现网络图片的读取与下载

    //网络图片的下载,读取与删除 public static void fileDowAndDel(String httpurl){ try { URL url = new URL(httpurl); ...

  9. Java学习笔记36(File类)

    File类可以对操作系统中的文件进行操作: File类的静态成员变量: package demo; import java.io.File; public class FileDemo { publi ...

  10. python 怎样获取toast?

    toast是什么? 想要获取toast的小伙伴们,肯定知道这个是一个什么玩意,例行还是加一个图,加以解释,下图的就是传说中的toast,它有一个特点,出现时间特别短,很难通过定位元素去获取这个toas ...