http://uoj.ac/problem/29

cdq四次处理出一直向左, 一直向右, 向左后回到起点, 向右后回到起点的dp数组,最后统计答案。

举例:\(fi\)表示一直向右走i天能参观的最多景点数。

其中有一个很重要的条件\(fi≤fi+1fi≤fi+1\),这个条件是分治的前提。

关于这个条件的证明,我想了好久才想出来,用反证法证明一下就行。

分治时需要用主席树维护路径上的前k大和。

#include<cstdio>
#include<cstring>
#include<algorithm>
#include"holiday.h"
using namespace std;
typedef long long ll;
const int N = 100003;
const int M = 250003; struct node {
int l, r, s;
ll sum;
node() {l = r = s = sum = 0;}
} T[N * 20];
int a[N], H[N], cnt = 0, root[N], top, st;
ll f[M], g[M], f1[M], g1[M]; void update(int &pos, int l, int r, int key) {
T[++cnt] = T[pos]; pos = cnt;
++T[pos].s; T[pos].sum += H[key];
if (l == r) return;
int mid = (l + r) >> 1;
if (key <= mid) update(T[pos].l, l, mid, key);
else update(T[pos].r, mid + 1, r, key);
} ll query(int tl, int tr, int l, int r, int num) {
if (l == r) {return min(T[tr].sum - T[tl].sum, 1ll * H[l] * num);}
int mid = (l + r) >> 1, s = T[T[tr].r].s - T[T[tl].r].s;
if (s >= num)
return query(T[tl].r, T[tr].r, mid + 1, r, num);
else
return T[T[tr].r].sum - T[T[tl].r].sum + query(T[tl].l, T[tr].l, l, mid, num - s);
} void cdq_f(int l, int r, int tmp_l, int tmp_r) {
if (l > r) return;
int mid = (l + r) >> 1, pos = tmp_l;
ll t;
for(int i = tmp_l; i - st <= mid && i <= tmp_r; ++i)
if ((t = query(root[st - 1], root[i], 1, top, mid - i + st)) > f[mid])
f[mid] = t, pos = i;
cdq_f(l, mid - 1, tmp_l, pos);
cdq_f(mid + 1, r, pos, tmp_r);
} void cdq_f1(int l, int r, int tmp_l, int tmp_r) {
if (l > r || tmp_l > tmp_r) return;
int mid = (l + r) >> 1, pos = tmp_l;
ll t;
for(int i = tmp_l; ((i - st) << 1) <= mid && i <= tmp_r; ++i)
if ((t = query(root[st], root[i], 1, top, mid - ((i - st) << 1))) > f1[mid])
f1[mid] = t, pos = i;
cdq_f1(l, mid - 1, tmp_l, pos);
cdq_f1(mid + 1, r, pos, tmp_r);
} void cdq_g(int l, int r, int tmp_l, int tmp_r) {
if (l > r) return;
int mid = (l + r) >> 1, pos = tmp_r;
ll t;
for(int i = tmp_r; st - i <= mid && i >= tmp_l; --i) {
if ((t = query(root[i - 1], root[st], 1, top, mid - st + i)) > g[mid])
g[mid] = t, pos = i;
}
cdq_g(l, mid - 1, pos, tmp_r);
cdq_g(mid + 1, r, tmp_l, pos);
} void cdq_g1(int l, int r, int tmp_l, int tmp_r) {
if (l > r || tmp_l > tmp_r) return;
int mid = (l + r) >> 1, pos = tmp_r;
ll t;
for(int i = tmp_r; ((st - i) << 1) <= mid && i >= tmp_l; --i) {
if ((t = query(root[i - 1], root[st - 1], 1, top, mid - ((st - i) << 1))) > g1[mid])
g1[mid] = t, pos = i;
}
cdq_g1(l, mid - 1, pos, tmp_r);
cdq_g1(mid + 1, r, tmp_l, pos);
} ll findMaxAttraction(int n, int start, int d, int attraction[]) {
for(int i = 0; i < n; ++i) H[++cnt] = attraction[i];
sort(H + 1, H + cnt + 1);
cnt = unique(H + 1, H + cnt + 1) - H;
for(int i = 1; i <= n; ++i) a[i] = lower_bound(H + 1, H + cnt, attraction[i - 1]) - H;
top = cnt - 1; cnt = 0; for(int i = 1; i <= n; ++i) {
root[i] = root[i - 1];
update(root[i], 1, top, a[i]);
} st = start + 1;
cdq_f(0, d, st, n);
cdq_f1(0, d, st + 1, n);
cdq_g(0, d, 1, st);
cdq_g1(0, d, 1, st - 1); ll ans = max(f[d], g[d]);
for(int i = 1; i <= d; ++i)
ans = max(ans, max(f[i] + g1[d - i], g[i] + f1[d - i]));
return ans;
}

UOJ上是交互题的形式

【UOJ #29】【IOI 2014】holiday的更多相关文章

  1. JAVA 基础编程练习题29 【程序 29 求矩阵对角线之和】

    29 [程序 29 求矩阵对角线之和] 题目:求一个 3*3 矩阵对角线元素之和 程序分析:利用双重 for 循环控制输入二维数组,再将 a[i][i]累加后输出. package cskaoyan; ...

  2. uoj 41 【清华集训2014】矩阵变换 婚姻稳定问题

    [清华集训2014]矩阵变换 Time Limit: 20 Sec  Memory Limit: 256 MB 题目连接 http://uoj.ac/problem/41 Description 给出 ...

  3. AC日记——【清华集训2014】奇数国 uoj 38

    #38. [清华集训2014]奇数国 思路: 题目中的number与product不想冲: 即为number与product互素: 所以,求phi(product)即可: 除一个数等同于在模的意义下乘 ...

  4. [官方软件] Easy Sysprep v4.3.29.602 【系统封装部署利器】(2016.01.22)--skyfree大神

    [官方软件] Easy Sysprep v4.3.29.602 [系统封装部署利器](2016.01.22) Skyfree 发表于 2016-1-22 13:55:55 https://www.it ...

  5. 【UOJ】67 新年的毒瘤 &【BZOJ】1123 BLO

    [UOJ 67] 题目链接: 传送门 题解: 第一眼很懵逼……这什么鬼. 思考什么点复合条件……(o(>﹏<)o 1.树,也就是说还剩n-2条边,等价于要删去一个度数为m-n+2的点. 2 ...

  6. 【UOJ#236】[IOI2016]railroad(欧拉回路,最小生成树)

    [UOJ#236][IOI2016]railroad(欧拉回路,最小生成树) 题面 UOJ 题解 把速度看成点,给定的路段看成边,那么现在就有了若干边,然后现在要补上若干边,以及一条\([inf,\) ...

  7. 【UOJ#177】欧拉回路

    [UOJ#177]欧拉回路 题面 UOJ 题解 首先图不连通就没啥好搞的了. 对于无向图而言,每个点度数为偶数. 对于有向图而言,每个点入度等于出度. 然后就是一本通上有的做法,直接\(dfs\)一遍 ...

  8. 【UOJ#311】【UNR #2】积劳成疾(动态规划)

    [UOJ#311][UNR #2]积劳成疾(动态规划) UOJ Solution 考虑最大值分治解决问题.每次枚举最大值所在的位置,强制不能跨过最大值,左右此时不会影响,可以分开考虑. 那么设\(f[ ...

  9. 【UOJ#450】【集训队作业2018】复读机(生成函数,单位根反演)

    [UOJ#450][集训队作业2018]复读机(生成函数,单位根反演) 题面 UOJ 题解 似乎是\(\mbox{Anson}\)爷的题. \(d=1\)的时候,随便怎么都行,答案就是\(k^n\). ...

随机推荐

  1. 2014 UESTC暑前集训动态规划专题解题报告

    A.爱管闲事 http://www.cnblogs.com/whatbeg/p/3762733.html B.轻音乐同好会 C.温泉旅馆 http://www.cnblogs.com/whatbeg/ ...

  2. Maven学习(九)插件介绍

    我们都知道Maven本质上是一个插件框架,它的核心并不执行任何具体的构建任务,所有这些任务都交给插件来完成,例如编译源代码是由maven-compiler-plugin完成的.进一步说,每个任务对应了 ...

  3. HTML 学习笔记 JavaScript (DOM)

    一 DOM 简介 通过HTML DOM 可以访问JavaScript 文档的所有元素 当网页被加载的时候,浏览器会创建页面的文档对象模型 HTML DOM 模型被构造成对象的树 HTML DOM 树 ...

  4. js 增删改查方法

    push() 向数组的末尾添加一个或多个元素 pop() 删除数组内部并返回数组的最后一个元素 shift() 把数组内部的第一个元素从其中删除,并返回第一个元素的值 unshift() 向数组外部的 ...

  5. 多态、GC、Java数据类型

    多态 一.java中实现多态的机制是什么? 靠的是: 父类定义的引用变量可以指向子类的实例对象,或者接口定义的引用变量可以指向具体实现类的实例对象 而程序调用的方法,在运行期才动态绑定, 它就是引用变 ...

  6. Theano2.1.1-基础知识之准备工作

    来源:http://deeplearning.net/software/theano/tutorial/index.html#tutorial 这里介绍的是使用theano的一些基础知识,虽然thea ...

  7. 简单高效的nodejs爬虫模型

    这篇文章讲解一下yunshare项目的爬虫模型. 使用nodejs开发爬虫很简单,不需要类似python的scrapy这样的爬虫框架,只需要用request或者superagent这样的http库就能 ...

  8. Oracle中序列(SEQUENCE)的使用一例

    曾经在触发器中使用序列(SEQUENCE): create or replace trigger TRI_SUPPLIER before insert on SUPPLIER for each row ...

  9. Python3.5 + django1.8.5 安装”import pymysql pymysql.install_as_MySQLdb()”的解决方法

    最近在学习Python,打算先看两个在线教程,再在github上找几个开源的项目练习一下,在学到"被解放的姜戈"时遇到django同步数据库时无法执行的错误,记录一下. 错误现象: ...

  10. nios II--实验3——led 100M硬件部分

    led_100M 硬件开发 新建原理图 参照实验二(led) QSys调用模块 参照实验二(led) 原理图添加IP模块 参照实验二(led),在调用PLL的时候需要修改系统和SDRAM的时钟频率为1 ...