题面传送门

题意:

有 \(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——伤害数字显示HUD

    伤害数字显示HUD 游戏中收到伤害掉血,会有飘动的伤害数值: 可以使用OnGUI中GUI.Label来实现: 可自定义字体,颜色,大小等: 如果需要更好看的数字特效,可以手动添加: 普通字体不够好看可 ...

  2. Golang通脉之错误处理

    在实际工程项目中,总是通过程序的错误信息快速定位问题,但是又不希望错误处理代码写的冗余而又啰嗦.Go语言没有提供像Java.C#语言中的try...catch异常处理方式,而是通过函数返回值逐层往上抛 ...

  3. 简明教程 | Docker篇 · 其二:Dockerfile的编写

    Dockerfile是什么 一个包含用于组合 image 的命令的文本文件,docker 通过 dockerfile 和构建环境的上下文来构建 image . 编写Dockerfile FROM 首先 ...

  4. Noip模拟40 2021.8.15

    T1 送花 按照题解意思说是扫描线题,但我觉得像一个线段树优化$dp$ 主要思想一样,就是暴力枚举右端点,同时维护左端点的最值, 考虑两种情况, 如果左端点在$r$扫到的数$i$上一次出现的位置之前, ...

  5. 三极管和MOS管驱动电路的正确用法

    1 三极管和MOS管的基本特性 三极管是电流控制电流器件,用基极电流的变化控制集电极电流的变化.有NPN型三极管(简称P型三极管)和PNP型三极管(简称N型三极管)两种,符号如下: MOS管是电压控制 ...

  6. Vue | uni-app 中使用websocket

    @ 目录 首先在根目录下新建一个store文件夹,并新建一个websocket.js文件,代码如下: import Vue from 'vue' import Vuex from 'vuex' Vue ...

  7. 15个问题自查你真的了解java编译优化吗?

    摘要:为什么C++的编译速度会比java慢很多?二者运行程序的速度差异在哪? 了解了java的早期和晚期过程,就能理解这个问题了. 本文分享自华为云社区<你真的了解java编译优化吗?15个问题 ...

  8. python mysqlclient安装失败 Command "python setup.py egg_info" failed with error code 1

    python2 python3 中代码 pip install mysqlclient 都安装失败的话, 很有可能是你的操作系统中没有安装mysql 如果确定已经安装了,请忽略下面的内容. Ubunt ...

  9. Python matplotlib pylot和pylab的区别

    matplotlib是Python中强大的绘图库. matplotlib下pyplot和pylab均可以绘图. 具体来说两者的区别 pyplot 方便快速绘制matplotlib通过pyplot模块提 ...

  10. 高度最小的BST 牛客网 程序员面试金典 C++ Python

    高度最小的BST 牛客网 程序员面试金典 C++ Python 题目描述 对于一个元素各不相同且按升序排列的有序序列,请编写一个算法,创建一棵高度最小的二叉查找树. 给定一个有序序列int[] val ...