题面传送门

题意:

有 \(n\) 个展销会,每个展销会给出它的时间 \(t_i\),举办展销会的位置 \(l_i\),和参加这个展销会你能得到的收益 \(m_i\)。

你现在在位置 \(s\),你可以花 \(u\) 的代价从 \(i\) 走到 \(i-1\),或是花 \(d\) 的代价从 \(i\) 走到 \(i+1\)。

你只能先参加时间靠前的展销会,再参加时间靠后的展销会。

问从 \(s\) 出发并回到 \(s\) 能够获得的最大收益。

\(1 \leq n,t_i \leq 5 \times 10^5\),\(1 \leq l_i \leq 5 \times 10^5+1\),可能存在相同的 \(t_i\)。

这是一道 dp 题。

先考虑一个弱化版本的问题,假设没有 \(t_i\) 相同。

不难想到将这 \(n\) 个展销会按 \(t_i\) 从小到大排个序。

然后设 \(dp_i\) 表示参加前 \(i\) 个展销会,当前在第 \(i\) 个展销会的最大收益。

枚举上一个参加的展销会,于是我们就有 \(dp_i=\max\{dp_j-cost(j,i)\}+m_i\),其中 \(cost(j,i)\) 表示从第 \(j\) 个展销会所在的位置到达第 \(i\) 个展销会所在的位置的代价。

我们进一步对 \(cost(j,i)\) 进行展开,就有

\(dp_i=\max\{dp_j-(l_j-l_i)\times u\}+m_i\quad(l_j\geq l_i)\)

\(dp_i=\max\{dp_j-(l_i-l_j)\times d\}+m_i\quad(l_j\lt l_i)\)

拿第一个柿子举例,将 \(dp\) 柿子拆成与 \(i\) 有关的和与 \(j\) 有关的两部分,也就有

\(dp_j-(l_j-l_i)\times u=dp_j-l_j\times u+l_i\times u=(dp_j-l_j\times u)+l_i\times u\)

用树状数组维护 \(dp_j-l_j\times u\) 的最大值就可以在 \(\mathcal O(\log n)\) 的时间内求出 \(dp_i\)


但加上 \(t_i\) 可能相同的条件怎么办呢?

在举办时间相同的展销会当中,我们肯定是从左往右依次经过某些展销会,或是从右到左依次经过某些展销会,不走回头路。

于是我们可以把这 \(t_i\) 相同的展销会按 \(l_i\) 从小到大排个序。(显然这 \(t_i\) 相同的展销会的下标能够组成一个区间)

先求出从举办时间小于 \(t_i\) 的展销会到达 \(i\) 的最小代价 \(x_i\)。

然后设 \(dpl_i\) 表示以 \(i\) 结尾的一条从左至右的路径的最小代价。

\(dpr_i\) 表示以 \(i\) 结尾的一条从右至左的路径的最小代价。

以 \(dpl_i\) 为例,要么从从举办时间小于 \(t_i\) 的展销会到达展销会 \(i\),也就是 \(x_i\),要么从展销会 \(i-1\) 到底 \(i\),也就是 \(dp_{i-1}-(l_i-l_{i-1})\times d+m_i\),两者取一个 \(\max\) 即可。

最后 \(dp_i=\max(dpl_i,dpr_i)\)。

别忘了最后要回到位置 \(s\)。

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int MAXN=500000+5;
const int MAXD=500001+5;
void chkmin(ll &x,ll y){(x>y)?(x=y):0;}
void chkmax(ll &x,ll y){(x<y)?(x=y):0;}
int n,s;ll u,d;
struct fair{
int t;ll l,m;
friend bool operator <(fair x,fair y){
if(x.t!=y.t) return x.t<y.t;
return x.l<y.l;
}
} a[MAXN];
struct fenwick_tree{
ll t[MAXD];
fenwick_tree(){memset(t,128,sizeof(t));}
void add(int x,ll v){for(int i=x;i<MAXD;i+=(i&(-i))) chkmax(t[i],v);}
ll query(int x){ll ret=-1e18;for(int i=x;i;i-=(i&(-i))) chkmax(ret,t[i]);return ret;}
} t1,t2;
ll dp[MAXN],dpd[MAXN],dpu[MAXN];
int main(){
scanf("%d%lld%lld%d",&n,&u,&d,&s);
t1.add(s,s*d);t2.add(MAXD-s+1,-s*u);
for(int i=1;i<=n;i++) scanf("%d%lld%lld",&a[i].t,&a[i].l,&a[i].m);
sort(a+1,a+n+1);memset(dp,128,sizeof(dp));
int l=1,r=1;
while(l<=n){
r=l;while(r<=n&&a[r].t==a[l].t) r++;r--;
// printf("%d %d\n",l,r);
for(int i=l;i<=r;i++){
dp[i]=max(dp[i],t1.query(a[i].l)-a[i].l*d);
dp[i]=max(dp[i],t2.query(MAXD-a[i].l-1)+a[i].l*u);
dp[i]+=a[i].m;
dpd[i]=dpu[i]=dp[i];
}
for(int i=l+1;i<=r;i++) dpd[i]=max(dpd[i],dpd[i-1]-(a[i].l-a[i-1].l)*d+a[i].m);
for(int i=r-1;i>=l;i--) dpu[i]=max(dpu[i],dpu[i+1]-(a[i+1].l-a[i].l)*u+a[i].m);
for(int i=l;i<=r;i++) dp[i]=max(dpd[i],dpu[i]);
for(int i=l;i<=r;i++) t1.add(a[i].l,dp[i]+a[i].l*d);
for(int i=l;i<=r;i++) t2.add(MAXD-a[i].l,dp[i]-a[i].l*u);
l=r+1;
}
// for(int i=1;i<=n;i++) printf("%lld\n",dp[i]);
ll ans=0;
for(int i=1;i<=n;i++) ans=max(ans,dp[i]-((a[i].l>s)?(a[i].l-s)*u:(s-a[i].l)*d));
printf("%lld\n",ans);
return 0;
}
/*
dp[i]=max{dp[j]+l[j]*d-l[i]*d}(l[j]<=l[i])
dp[i]=max{dp[j]-l[j]*u+l[i]*u}(l[j]>l[i])
*/

洛谷 P5902 [IOI2009]salesman(dp)的更多相关文章

  1. 洛谷P1108 低价购买[DP | LIS方案数]

    题目描述 “低价购买”这条建议是在奶牛股票市场取得成功的一半规则.要想被认为是伟大的投资者,你必须遵循以下的问题建议:“低价购买:再低价购买”.每次你购买一支股票,你必须用低于你上次购买它的价格购买它 ...

  2. NOIP2017提高组Day2T2 宝藏 洛谷P3959 状压dp

    原文链接https://www.cnblogs.com/zhouzhendong/p/9261079.html 题目传送门 - 洛谷P3959 题目传送门 - Vijos P2032 题意 给定一个 ...

  3. 洛谷P1244 青蛙过河 DP/思路

    又是一道奇奇怪怪的DP(其实是思路题). 原文戳>>https://www.luogu.org/problem/show?pid=1244<< 这题的意思给的挺模糊,需要一定的 ...

  4. 洛谷P3928 Sequence2(dp,线段树)

    题目链接: 洛谷 题目大意在描述底下有.此处不赘述. 明显是个类似于LIS的dp. 令 $dp[i][j]$ 表示: $j=1$ 时表示已经处理了 $i$ 个数,上一个选的数来自序列 $A[0]$ 的 ...

  5. 洛谷P1140 相似基因 (DP)

    洛谷P1140 相似基因 题目背景 大家都知道,基因可以看作一个碱基对序列.它包含了44种核苷酸,简记作A,C,G,TA,C,G,T.生物学家正致力于寻找人类基因的功能,以利用于诊断疾病和发明药物. ...

  6. 洛谷P2224 [HNOI2001] 产品加工 [DP补完计划,背包]

    题目传送门 产品加工 题目描述 某加工厂有A.B两台机器,来加工的产品可以由其中任何一台机器完成,或者两台机器共同完成.由于受到机器性能和产品特性的限制,不同的机器加工同一产品所需的时间会不同,若同时 ...

  7. 洛谷1417 烹调方案 dp 贪心

    洛谷 1417 dp 传送门 挺有趣的一道dp题目,看上去接近于0/1背包,但是考虑到取每个点时间不同会对最后结果产生影响,因此需要进行预处理 对于物品x和物品y,当时间为p时,先加x后加y的收益为 ...

  8. 洛谷1387 二维dp 不是特别简略的题解 智商题

    洛谷1387 dp题目,刚开始写的时候使用了前缀和加搜索,复杂度大概在O(n ^ 3)级别,感觉这么写还是比较对得起普及/提高-的难度的..后来看了题解区各位大神的题解,开始一脸mb,之后备受启发. ...

  9. 洛谷 P2657 (数位DP)

    ### 洛谷 P2657 题目链接 ### 题目大意:给你一个数的范围 [A,B] ,问你这段区间内,有几个数满足如下条件: 1.两个相邻数位上的数的差值至少为 2 . 2.不包含前导零. 很简单的数 ...

随机推荐

  1. Unity——EasyTouch摇杆插件使用

    EasyTouch摇杆插件使用 Demo展示 双指缩放在电脑端无法掩饰,竖屏将就看看吧: 插件名叫EasyTouch,有需要给我留言,不想开仓库传了: 创建摇杆点这里: 初始化 On_Joystick ...

  2. UltraSoft - DDL Killer - Alpha 项目展示

    团队介绍 CookieLau fmh 王 FUJI LZH DZ Monster PM & 后端 前端 前端 前端 后端 后端 软件介绍 项目简介 项目名称:DDLKiller 项目描述:&q ...

  3. [BUAA]起点 软工第一次作业-热身

    项目 内容 这个作业属于哪个课程 2020计算机学院软件工程(罗杰 任健) 这个作业的要求在哪里 第一次作业-热身! 我在这个课程的目标是 了解软件开发,提高自己的工程能力和团队协作能力 这个作业在哪 ...

  4. 零基础入门非常好的C语言基础资料

    C语言程序的结构认识 用一个简单的c程序例子,介绍c语言的基本构成.格式.以及良好的书写风格,使小伙伴对c语言有个初步认识. 例1:计算两个整数之和的c程序: #include main() { in ...

  5. 零基础入门之Linux进程基础

    计算机实际上可以做的事情实质上非常简单,比如计算两个数的和,再比如在内存中寻找到某个地址等等.这些最基础的计算机动作被称为指令(instruction).所谓的程序(program),就是这样一系列指 ...

  6. 洛谷 P3147 [USACO16OPEN]262144 P

    链接: P3147 P3146双倍经验 前言: 今天发现的一道很有意思的DP题 分析: 第一眼以为是区间DP,于是设f[i][j]为从第i个数到第j个数可以合出的最大值,但思考后发现并不能简单合并,并 ...

  7. C++构造函数注意事项

    1.匿名对象 首先应该明确匿名对象,匿名对象是之没有对象名,调用完构造函数后即析构的对象.下面通过代码捕捉类的构造函数和析构函数,以进行说明: #include <iostream> us ...

  8. 安装与卸载JDK8

    前言:学习Java的第一步需要先配置好JDK环境,而JDK8是目前使用最广泛的JDK版本.本文讲解了如何下载安装和卸载JDK8.以下环境为Windows10 下载JDK安装包 Oracle官网 所有J ...

  9. Windows 2008 R2 NTP 时钟同步配置

    一.配置 本地组策略 a.windows+R 或  "开始菜单"  | "运行"  ,打开运行窗口. 输入gpedit.msc打开本地组策略 b.在 本地组策略 ...

  10. Zabbix5.0实现监控系统登陆失败告警

    环境zabbix5.0,配置思路,通过添加监控项和触发器实现,监控项监控对应的日志文件,触发器过滤日志文件中的关键字,当出现failed时就发出告警. 监控项配置 类型选择zabbix客户端主动式,键 ...