「JSOI2010」挖宝藏

传送门

由于题目中说道挖一个位置的前提是挖掉它上面的三个,以此类推可以发现,挖掉一个点就需要挖掉这个点往上的整个倒三角,那么也就会映射到 \(x\) 轴上的一段区间(可以发现这种映射关系是一一对应的),那么我们就可以用一段区间来代表一个宝藏。

然后我们就先把所有区间按照右端点递增其次左端点递增排序。

接着考虑 \(\text{DP}\) ,我们设 \(dp_i\) 表示前 \(i\) 个区间中强制选第 \(i\) 个区间的最大利润,

那么在枚举转移点 \(j\) 时就会出现 \(i\) ,\(j\) 有交的情况,就会有一部分代价被多算。

此外还有一种情况就是一个区间完全覆盖另一个的情况,这个时候如果我们选那个较大的区间肯定会顺带选了那个较小的,

因为此时那个较小区间的代价就不用算了,所以我们可以预处理出单选一个区间的最大利润(它自己的价值以及被它覆盖的所有区间的价值之和 - 它自己的代价),

但我们又会发现,转移时会出现 \(i\) , \(j\) 两个区间同时覆盖一个小区间,导致那个小区间的价值被算重的情况,

所以我们干脆对于两个区间有交的情况我们暴力地去算可能被算重的价值即可,

具体来说就是枚举到 \(i\) 的时候,维护一个指针指向我们需要计算的区间,因为这个指针只会右移,所以我们转移的复杂度还是 \(O(n)\) 的,总复杂度也就是 \(O(n^2)\) 的。

如果有不懂的可以结合代码理解,还可以画图自己研究研究。

参考代码:

#include <algorithm>
#include <cstdio>
#define rg register
#define file(x) freopen(x".in", "r", stdin), freopen(x".out", "w", stdout)
using namespace std;
template < class T > inline void read(T& s) {
s = 0; int f = 0; char c = getchar();
while ('0' > c || c > '9') f |= c == '-', c = getchar();
while ('0' <= c && c <= '9') s = s * 10 + c - 48, c = getchar();
s = f ? -s : s;
} const int _ = 1002; int n, dp[_]; struct node { int l, r, p1, p2, c; } t[_];
inline bool cmp(const node& x, const node& y) { return x.r != y.r ? x.r < y.r : x.l < y.l; } inline int calc(int l, int r) { return (r - l + 2) * (r - l + 2) / 4; } int main() {
#ifndef ONLINE_JUDGE
file("cpp");
#endif
read(n);
for (rg int x, y, p, i = 1; i <= n; ++i)
read(x), read(y), read(p), t[i] = (node) { x + y + 1, x - y - 1, p, 0, calc(x + y + 1, x - y - 1) };
for (rg int i = 1; i <= n; ++i)
for (rg int j = 1; j <= n; ++j)
if (t[i].l <= t[j].l && t[j].r <= t[i].r) t[i].p2 += t[j].p1;
sort(t + 1, t + n + 1, cmp);
for (rg int i = 1; i <= n; ++i) {
dp[i] = t[i].p2 - t[i].c;
int nxt = 1, sum = 0;
for (rg int j = 1; j < i; ++j) {
if (t[j].r < t[i].l) dp[i] = max(dp[i], dp[j] + t[i].p2 - t[i].c);
if (t[j].l < t[i].l && t[i].l <= t[j].r) {
while (nxt <= i && t[nxt].r <= t[j].r) {
if (t[nxt].l >= t[i].l) sum += t[nxt].p1; ++nxt;
}
dp[i] = max(dp[i], dp[j] + t[i].p2 - t[i].c - (sum - calc(t[i].l, t[j].r)));
}
}
}
int ans = 0;
for (rg int i = 0; i <= n; ++i) ans = max(ans, dp[i]);
printf("%d\n", ans);
return 0;
}

「JSOI2010」挖宝藏的更多相关文章

  1. 「JSOI2010」排名

    「JSOI2010」排名 传送门 看到先后顺序限制和字典序,很容易想到拓扑排序 + 贪心. 考虑具体做法: 对于第一问: 我们开一个大根堆来代替队列,然后从大到小构造出各个元素的排名. 我们连边 \( ...

  2. 「JSOI2010」找零钱的洁癖

    「JSOI2010」找零钱的洁癖 传送门 个人感觉很鬼的一道题... 首先我们观察到不同的数最多 \(50\) 个,于是考虑爆搜. 但是这样显然不太对啊,状态数太多了. 然后便出现了玄学操作: \(\ ...

  3. 「JSOI2010」旅行

    「JSOI2010」旅行 传送门 比较妙的一道 \(\text{DP}\) 题,思维瓶颈应该就是如何确定状态. 首先将边按边权排序. 如果我们用 \(01\) 串来表示 \(m\) 条边是否在路径上, ...

  4. 【LOJ】 #2520. 「FJOI2018」所罗门王的宝藏

    题解 发现似乎相当于问一个2000个元的方程组有没有解-- 然而我懵逼啊-- 发现当成图论,两个点之间连一条边,开始BFS,每个点的值赋成边权减另一个点的点权 如果一个环不合法那么肯定无解 代码 #i ...

  5. LG4171/BZOJ1823 「JSOI2010」满汉全席 2-SAT

    问题描述 LG4171 BZOJ1823 题解 显然,每个评委对每个材料的满式/汉式要求是对\(n\)个元素的\(0,1\)取值限制. 显然想到\(\mathrm{2-SAT}\) 于是就可以切掉了. ...

  6. 「JSOI2010」满汉全席

    前言 由于蒟蒻才刚开始学 \(\text{2-SAT}\),所以题解中有的地方可能不够精炼,望多包涵! 题目描述 题目意思很简单,标准的\(\text{2-SAT}\)问题模型.那么我们就先来介绍一下 ...

  7. 「BZOJ1924」「SDOI2010」 所驼门王的宝藏 tarjan + dp(DAG 最长路)

    「BZOJ1924」[SDOI2010] 所驼门王的宝藏 tarjan + dp(DAG 最长路) -------------------------------------------------- ...

  8. 「NOIP2017」宝藏

    「NOIP2017」宝藏 题解 博客阅读效果更佳 又到了一年一度NOIPCSP-S 赛前复习做真题的时间 于是就遇上了这道题 首先观察数据范围 \(1 \le n \le 12\) ,那么极大可能性是 ...

  9. loj #2037. 「SHOI2015」脑洞治疗仪

    #2037. 「SHOI2015」脑洞治疗仪   题目描述 曾经发明了自动刷题机的发明家 SHTSC 又公开了他的新发明:脑洞治疗仪——一种可以治疗他因为发明而日益增大的脑洞的神秘装置. 为了简单起见 ...

随机推荐

  1. static的使用总结

    全局静态变量 全局变量前加上关键字static,全局变量就定义成一个全局静态变量.,全局静态变量存储在静态存储区,在整个程序运行期间一直存在.全局静态变量在程序运行之前就存在. 初始化:未经初始化的全 ...

  2. C#堆和栈的入门理解

    声明:以下内容从网络整理,非原创,适当待入个人理解. 解释1.栈是编译期间就分配好的内存空间,因此你的代码中必须就栈的大小有明确的定义:堆是程序运行期间动态分配的内存空间,你可以根据程序的运行情况确定 ...

  3. Windows引用opencv静态库

    参考博客:https://www.cnblogs.com/sysuzyq/p/6183568.html

  4. Java读取、写入、处理Excel文件中的数据(转载)

    原文链接 在日常工作中,我们常常会进行文件读写操作,除去我们最常用的纯文本文件读写,更多时候我们需要对Excel中的数据进行读取操作,本文将介绍Excel读写的常用方法,希望对大家学习Java读写Ex ...

  5. 2019冬季PAT甲级第三题

    #define HAVE_STRUCT_TIMESPEC #include<bits/stdc++.h> using namespace std; ]; ]; ]; int main(){ ...

  6. Java代码三级跳——表达式、语句和代码块

    Java代码三级跳—表达式.语句和代码块 表达式(expression):Java中最基本的一个运算.比如一个加法运算表达式.1+2是一个表达式,a+b也是. 语句(statement):类似于平时说 ...

  7. Tarjan-有向图

    (我到底是咕了多少知识点啊) 在有向图中tarjan主要用来求强连通分量并缩点 一.定义 强连通:如果两个顶点可以相互通达,则称两个顶点 强连通 强连通分量:如果有向图G的每两个顶点都 强连通,称G是 ...

  8. 6_6 小球下落(UVa679)<完全二叉树编号>

    有K个球从一完整二叉树(fully binary tree,FBT)的树根(root)一个一个往下掉.当这个球遇到非终端节点时,可能往左子树跑,也可能往右子树跑,如此直到这颗球到达终端节点(也就是树叶 ...

  9. 安卓开发:Android Studio自动import

    我只想说,真好用!哈哈,提高效率的好东西. 参考: [https://blog.csdn.net/pjdd123/article/details/80953669] [https://www.cnbl ...

  10. ‘\0’的ASCII码

    1.'\0'的ASCII码为0 2.用串口发送字符串时,可以通过'\0'判断字符串是否结束,但发送数字数组的时候不能通过'\0'判断数组是否结束,因为数字0与'\0'的ASCII码值相同.