计划着刷\(DP\)题结果碰到了这样一道论文题,幸好不是太难。


【题目大意】

口水话有点多,所以就直接放链接。传送门

【分析】

看到题首先联想到了曾经做过的关路灯。所以先按\(x\)值排序,然后进行区间\(DP\)。

不妨设\(f_1[i][j]\),\(f_2[i][j]\)分别表示从起点出发已射落\(i\)到\(j\)这一段彩蛋,当前停留在\(i\)点,\(j\)点的最大得分\(v\)。

考虑 \(f_1[i][j]\),即点\(i\)是当前射击的彩蛋,射击的得分与当前时刻挂钩,但 是当前的时刻是不能从\(f_1[i][j]\)的状态中表示出来的,我们进一步考虑 \(f_1[i][j]\)的求解。

由于射击\(i\)的得分是\(y_i−t∗v_i\),而\(t\)等于之前每一步决策移动的时间总和,这样我们就可以把\(t∗v_i\)​在之前的移动中就计算,也就是说每次移动都要把未来会减少的得分计算在内。 比如说从\(f_1[i+1][j]\)推到\(f_1[i][j]\),即从\(i+1\)走到\(i\)时除了\(i+1\)到\(j\)这一段彩蛋外,其它的彩蛋都在下落,将这丢失的分数一并计算到从\(i+1\)走到\(i\)中。由于\(-t*v_i\)已经在之前决策时计算,所以射击时直接加上\(y_i\)即可。

所以可以先用\(sum[]\)计算\(v_i\)的前缀和,然后\(DP\)方程:

\[f_1[i][j]=y[i]+max(f_1[i+1][j]+Sum(i+1,j)*(x_{i+1}-x_i),f_2[i+1][j]+Sum(i+1,j)*(x_j-x_i)
\]

\[f_2[i][j]=y[i]+max(f_1[i][j-1]+Sum(i,j-1)*(x_{j}-x_{i}),f_2[i][j-1]+Sum(i,j-1)*(x_j-x_{j-1})
\]

然后\(O(n^2)\)过。

【Code】

#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<algorithm>
#define ll long long
using namespace std;
const int MAX = 1000 + 5;
const int INF = 0x3f3f3f3f;
inline int read(){
int f = 1, x = 0;char ch;
do { ch = getchar(); if (ch == '-') f = -1; } while (ch < '0'||ch>'9');
do {x = x*10+ch-'0'; ch = getchar(); } while (ch >= '0' && ch <= '9');
return f*x;
}
int n, bj;
double x0, f[3][MAX][MAX], sum[MAX];
struct sakura { double x, y, v; }sak[MAX];
inline bool cmp(sakura a, sakura b) { return a.x < b.x; }
inline double Sum(int l, int r) { return sum[n] - sum[r] + sum[l - 1]; }
inline double ab(double a) { return a < 0 ? -a : a; }
int main(){
n = read(); ++n, x0 = read();
sak[1].x = x0, sak[1].y = sak[1].v = 0;
for (int i = 2;i <= n; ++i) {
sak[i].x = read();
}
for (int i = 2;i <= n; ++i) {
sak[i].y = read();
}
for (int i = 2;i <= n; ++i) {
sak[i].v = read();
}
sort(sak + 1, sak + 1 + n, cmp);
for (int i = 1;i <= n; ++i) {
sum[i] = sum[i - 1] + sak[i].v;
if (ab(sak[i].x - x0) <= 1e-10 && ab(sak[i].y) <= 1e-10) {
bj = i;
}
}
memset(f, -INF, sizeof (f));
f[1][bj][bj] = f[2][bj][bj] = 0.0;
for (int k = 1;k <= n; ++k) {
for (int i = 1;i + k <= n; ++i) {
int j = i + k;
f[1][i][j] = sak[i].y + max(f[1][i + 1][j] - (sak[i + 1].x - sak[i].x) * Sum(i + 1, j), f[2][i + 1][j] - (sak[j].x - sak[i].x) * Sum(i + 1, j));
f[2][i][j] = sak[j].y + max(f[1][i][j - 1] - (sak[j].x - sak[i].x) * Sum(i, j - 1), f[2][i][j -1] - (sak[j].x - sak[j - 1].x) * Sum(i, j - 1));
}
}
printf("%.3lf", max(f[1][1][n], f[2][1][n]) / 1000.0);
return 0;
}

后来听人说这是未来\(DP\)???

【简●解】[SDOI2008] Sue的小球的更多相关文章

  1. 【BZOJ2037】[Sdoi2008]Sue的小球 区间DP+费用提前

    [BZOJ2037][Sdoi2008]Sue的小球 Description Sue和Sandy最近迷上了一个电脑游戏,这个游戏的故事发在美丽神秘并且充满刺激的大海上,Sue有一支轻便小巧的小船.然而 ...

  2. 2037: [Sdoi2008]Sue的小球

    2037: [Sdoi2008]Sue的小球 链接 题解 论文 代码 #include<cstdio> #include<algorithm> #include<cstr ...

  3. BZOJ2037: [Sdoi2008]Sue的小球

    Description Sue 和Sandy最近迷上了一个电脑游戏,这个游戏的故事发在美丽神秘并且充满刺激的大海上,Sue有一支轻便小巧的小船.然而,Sue的目标并不是当一个海 盗,而是要收集空中漂浮 ...

  4. [SDOI2008]Sue的小球

    题目描述 Sue和Sandy最近迷上了一个电脑游戏,这个游戏的故事发在美丽神秘并且充满刺激的大海上,Sue有一支轻便小巧的小船.然而,Sue的目标并不是当一个海盗,而是要收集空中漂浮的彩蛋,Sue有一 ...

  5. [luogu2446][bzoj2037][SDOI2008]Sue的小球【区间DP】

    分析 简单区间DP, 定义状态f[i][j][0/1]为取完i-j的小球最后取i/j上的小球所能获得的最大价值. 排序转移. ac代码 #include <bits/stdc++.h> # ...

  6. BZOJ2037: [Sdoi2008]Sue的小球(区间DP)

    Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 869  Solved: 483[Submit][Status][Discuss] Description ...

  7. bzoj 2037: [Sdoi2008]Sue的小球——dp

    Description Sue和Sandy最近迷上了一个电脑游戏,这个游戏的故事发在美丽神秘并且充满刺激的大海上,Sue有一支轻便小巧的小船.然而,Sue的目标并不是当一个海盗,而是要收集空中漂浮的彩 ...

  8. Luogu[SDOI2008]Sue的小球

    题目描述 Sue和Sandy最近迷上了一个电脑游戏,这个游戏的故事发在美丽神秘并且充满刺激的大海上,Sue有一支轻便小巧的小船.然而,Sue的目标并不是当一个海盗,而是要收集空中漂浮的彩蛋,Sue有一 ...

  9. bzoj 2037: [Sdoi2008]Sue的小球

    #include<cstdio> #include<iostream> #include<algorithm> using namespace std; struc ...

随机推荐

  1. 洛谷 - P1801 - 黑匣子 - 对顶堆

    这道题是提高+省选-的难度,做出来的话对数据结构题目的理解会增加很多. 可以使用一种叫做对顶堆的东西,对顶堆是在线维护第n小的logn的算法.大概的思路是,假如我们要找的是第n小,我们就维护一个大小为 ...

  2. 百度之星资格赛 1003 度度熊与邪恶大魔王(二维dp)

    分析 挺好的一道题 dp[i][j]表示打败i颗血j防御力的怪兽需要的最少宝石数 然后就好了,复杂度\(O(n+m*1000*10)\) #include <bits/stdc++.h> ...

  3. CCF2016.4 - C题

    思路:先把路径按反斜杠split成数组,然后用一个ArrayList去模拟.如果遇到空或者.则不处理:如果遇到..则删除ArrayList最后一个元素(注意如果只有1个元素则不删除):其他情况直接加到 ...

  4. Python解释器的安装步骤

    Python是一门强大的语言,目前已支持所有主流操作系统,在Linux,Unix,Mac系统上自带Python环境,在Windows10系统上需要安装一下,超简单 1.  打开官网 https://w ...

  5. the little schemer 笔记(4)

    第四章 numbers games 14 是原子吗 是的,数都是原子 (atom? n) 是真还是假,其中n是14 真,14 是原子 -3是数吗 是的,不过我们暂不考虑负数 3.14159是数吗 是的 ...

  6. Jmeter之添加响应断言,bean shell post processor

    一直在想运用jmeter来实现接口自动化测试,但是每次每个接口执行完,需要肉眼去看一看,执行的结果对不对,总结了两种办法, 一.将每个请求的响应结果,导出到文件 选中请求右键-->添加后置处理器 ...

  7. Qt之对话框QDialog

    这一节主要讲述对话框类,先讲述两种不同类型的对话框,再介绍Qt提供的几个标准对话框.对应本节的内容,可以在帮助索引中查看 QDialog 和 Dialog Windows 关键字. 一.模态和非模态对 ...

  8. 洛谷P3379lca,HDU2586,洛谷P1967货车运输,倍增lca,树上倍增

    倍增lca板子洛谷P3379 #include<cstdio> struct E { int to,next; }e[]; ],anc[][],log2n,deep[],n,m,s,ne; ...

  9. Coloring Trees CodeForces - 711C

    Coloring Trees CodeForces - 711C 题意:有n个点,每个点有一个c值,如果为0表示它没有被染色,否则表示它被染成了c值的颜色.颜色有1到m.把第i棵树染成颜色j所需要的代 ...

  10. LCA最近公共祖先知识点整理

    题解报告:hdu 2586 How far away ? Problem Description There are n houses in the village and some bidirect ...