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. windows 性能监视器

    转载地址:https://www.cnblogs.com/luo-mao/p/5872374.html

  2. 牛客练习赛31 D 神器大师泰兹瑞与威穆 STL,模拟 A

    牛客练习赛31 D 神器大师泰兹瑞与威穆 https://ac.nowcoder.com/acm/contest/218/D 时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 26214 ...

  3. ABP框架系列之三十一:(Localization-本地化)

    Introduction Any application has at least one language for user interface. Many applications have mo ...

  4. (PMP)第12章-----项目采购管理

    B D 12.1 规划采购管理 输入 工具与技术 输出 1.项目章程 2.商业文件 (商业文件, 效益管理计划) 3.项目管理计划 (范围,质量,资源管理计划, 范围基准) 4.项目文件 (里程碑清单 ...

  5. Python开发——4.集合和字符串拼接

    一.集合(set) 1.集合的特性: 不同元素组成.元素是无序排列的可hash值 2.集合转为列表 s1 = {11,"hechouzi",(11,22,33)} names = ...

  6. 系统性能--CPU

    对于cpu,目前比较关心的是cpu的利用率还有cpu的load,或者还有cpu运行队列. cpu利用率 cpu利用率分为sys,us.分别为操作系统和用户进程所占用的cpu利用率.sys的占用,一般是 ...

  7. 我知道的nginx配置

    1.nginx配置文件 2.配置访问域名 #京淘商品管理系统 server { listen 80; server_name manage.jt.com; location / { proxy_pas ...

  8. WeexSDK之注册Modules

    注册Modules的流程和注册Components非常类似. + (void)_registerDefaultModules { [self registerModule:@"dom&quo ...

  9. 背水一战 Windows 10 (73) - 控件(控件基类): UIElement - 拖放的基本应用, 手动开启 UIElement 的拖放操作

    [源码下载] 背水一战 Windows 10 (73) - 控件(控件基类): UIElement - 拖放的基本应用, 手动开启 UIElement 的拖放操作 作者:webabcd 介绍背水一战 ...

  10. boost::bind 实现原理, 手动实现一个

    template<typename R, typename T, typename A1> class hangj_call { public: hangj_call(R (T::*f_) ...