这题的加强版,多了一个$l_i$的限制,少了一个$p_i$的单调性,难了好多...

  首先有方程$f(i)=min\{f(j)+(dep_i-dep_j)*p_i+q_i\}$

  $\frac {f(j)-f(k)}{dep_j-dep_k}<p_i$

  假如没有$l_i$的限制,实际上就是上面那题...

  如果多了$l_i$的限制会有什么影响呢?

  类似上图的情况...红线是$l_i$的限制,如果是单调队列写法的话,P点会被删掉,但实际上P点依然有可能成为最优决策点...

  这个时候一个单调队列的写法就不可行了...

  之所以会出现这个问题,是因为P点被不属于$l_i$范围内的点给弹出了,所以我们只能在不存在$l_i$之外的点的单调队列里查,但是这样我们要开$O(n)$个队列,显然无法承受,但是我们知道的是,凸包是可以合并的,所以直接用线段树来维护就好了。

  我们需要维护$2n$个队列,像线段树一样,分别维护深度为$[1,dep],[1,dep/2],[dep/2+1,dep],[1,dep/4],[dep/4+1,dep/2],...$的单调队列,每次询问的时候把在距离范围内的区间一个一个合并起来,每个区间都需要二分$O(logn)$找出最优决策点,最后合并$O(logn)$个区间,合并的时候只需要求出一个各个区间最优决策点的凸包就好了,每次合并$O(1)$,复杂度$O(nlog^2n)$。

  修改的话同理,把包含当前点的所有区间都二分找到出队位置移动队尾位置,同时记录下被覆盖的点,维护一下可持久化凸包,然后就可以了,这一部分细节比较多,结合代码更好理解。

  队列数组开小+很多地方没开LL+困=调了1h,第一次编译的时候代码就没有实质上的错误了,但是却调了那么久,sigh...

#include<iostream>
#include<cstring>
#include<cstdlib>
#include<cstdio>
#include<algorithm>
#define ll long long
#define int long long
using namespace std;
const int maxn=, inf=1e9;
struct poi{int too, dis, pre;}e[maxn];
struct tjm{int bg, len;}tree[maxn<<];
int n, m, x, z, tot, sum;
int v[maxn*], last[maxn], p[maxn], l[maxn], memv[maxn][], meml[maxn][];
ll f[maxn], dep[maxn], deep[maxn], q[maxn];
inline void read(int &k)
{
int f=; k=; char c=getchar();
while(c<'' || c>'') c=='-'&&(f=-), c=getchar();
while(c<='' && c>='') k=k*+c-'', c=getchar();
k*=f;
}
inline void add(int x, int y, int z){e[++tot]=(poi){y, z, last[x]}; last[x]=tot;}
inline double xl(int x, int y){return 1.0*(f[x]-f[y])/(deep[x]-deep[y]);}
inline int findans(int l, int r, ll limit)
{
if(l>=r) return r;
int L=l+, R=r;
while(L<R)
{
int mid=(L+R)>>;
if(limit-xl(v[mid], v[mid-])>1e-) L=mid+;
else R=mid;
}
if(xl(v[L], v[L-])>=limit) L--;
return L;
}
inline int find(int r, int limit)
{
int L=, R=r;
while(L<R)
{
int mid=(L+R)>>;
if(dep[mid]>=limit) R=mid;
else L=mid+;
}
return L;
}
inline int findr(int l, int r, int pos)
{
if(l>=r) return r;
int L=l+, R=r;
while(L<R)
{
int mid=(L+R+)>>;
if(xl(v[mid], v[mid-])-xl(v[mid], pos)>1e-) R=mid-;
else L=mid;
}
if(xl(v[L], v[L-])-xl(v[L], pos)>1e-) L--;
return L;
}
void build(int x, int l, int r)
{
tree[x].bg=sum; sum+=r-l+;
if(l==r) return;
int mid=(l+r)>>;
build(x<<, l, mid); build(x<<|, mid+, r);
}
int query(int x, int l, int r, int cx, ll limit)
{
if(!tree[x].len || r<cx) return -;
if(cx<=l) return findans(tree[x].bg, tree[x].bg+tree[x].len-, limit);
int mid=(l+r)>>, lt=query(x<<, l, mid, cx, limit), rt=query(x<<|, mid+, r, cx, limit);
if(lt!=- && rt!=-) if(limit-xl(v[lt], v[rt])>1e-) swap(lt, rt);
return lt==-?rt:lt;
}
void update(int x, int l, int r, int cx, int pos, int d)
{
int R=findr(tree[x].bg, tree[x].bg+tree[x].len-, pos);
memv[pos][d]=v[++R]; meml[pos][d]=tree[x].len; v[R]=pos; tree[x].len=R-tree[x].bg+;
if(l==r) return;
int mid=(l+r)>>;
if(cx<=mid) update(x<<, l, mid, cx, pos, d+);
else update(x<<|, mid+, r, cx, pos, d+);
}
void recovey(int x, int l, int r, int cx, int pos, int d)
{
v[tree[x].bg+tree[x].len-]=memv[pos][d]; tree[x].len=meml[pos][d];
if(l==r) return;
int mid=(l+r)>>;
if(cx<=mid) recovey(x<<, l, mid, cx, pos, d+);
else recovey(x<<|, mid+, r, cx, pos, d+);
}
void dfs(int x, int d)
{
if(x!=)
{
int limit=dep[d]-l[x], nxt=v[query(, , n, find(d-, limit), p[x])];
f[x]=f[nxt]+(deep[x]-deep[nxt])*p[x]+q[x];
}
update(, , n, d, x, );
for(int i=last[x], too;i;i=e[i].pre)
deep[too=e[i].too]=deep[x]+e[i].dis, dep[d+]=dep[d]+e[i].dis, dfs(too, d+);
recovey(, , n, d, x, );
}
#undef int
int main()
{
read(n); read(m);
for(int i=;i<=n;i++)
read(x), read(z), read(p[i]), read(q[i]), read(l[i]), add(x, i, z);
sum=; build(, , n); dfs(, );
for(int i=;i<=n;i++) printf("%lld\n", f[i]);
}

bzoj3672: [Noi2014]购票(树形DP+斜率优化+可持久化凸包)的更多相关文章

  1. 洛谷P3994 Highway(树形DP+斜率优化+可持久化线段树/二分)

    有点类似NOI2014购票 首先有方程$f(i)=min\{f(j)+(dep_i-dep_j)*p_i+q_i\}$ 这个显然是可以斜率优化的... $\frac {f(j)-f(k)}{dep_j ...

  2. BZOJ_3672_ [Noi2014]购票_CDQ分治+斜率优化

    BZOJ_3672_ [Noi2014]购票_CDQ分治+斜率优化 Description  今年夏天,NOI在SZ市迎来了她30周岁的生日.来自全国 n 个城市的OIer们都会从各地出发,到SZ市参 ...

  3. [NOI2014]购票 「树上斜率优化」

    首先易得方程,且经过变换有 $$\begin{aligned} f_i &= \min\limits_{dist_i - lim_i \le dist_j} \{f_j + (dist_i - ...

  4. 【BZOJ-3672】购票 树分治 + 斜率优化DP

    3672: [Noi2014]购票 Time Limit: 30 Sec  Memory Limit: 512 MBSubmit: 1177  Solved: 562[Submit][Status][ ...

  5. 【BZOJ3672】[Noi2014]购票 树分治+斜率优化

    [BZOJ3672][Noi2014]购票 Description  今年夏天,NOI在SZ市迎来了她30周岁的生日.来自全国 n 个城市的OIer们都会从各地出发,到SZ市参加这次盛会.       ...

  6. P3994 高速公路 树形DP+斜率优化+二分

    $ \color{#0066ff}{ 题目描述 }$ C国拥有一张四通八达的高速公路网树,其中有n个城市,城市之间由一共n-1条高速公路连接.除了首都1号城市,每个城市都有一家本地的客运公司,可以发车 ...

  7. Codeforces 1179D 树形DP 斜率优化

    题意:给你一颗树,你可以在树上添加一条边,问添加一条边之后的简单路径最多有多少条?简单路径是指路径中的点只没有重复. 思路:添加一条边之后,树变成了基环树.容易发现,以基环上的点为根的子树的点中的简单 ...

  8. 【BZOJ-4518】征途 DP + 斜率优化

    4518: [Sdoi2016]征途 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 230  Solved: 156[Submit][Status][ ...

  9. 【BZOJ-3437】小P的牧场 DP + 斜率优化

    3437: 小P的牧场 Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 705  Solved: 404[Submit][Status][Discuss ...

随机推荐

  1. DNS分离解析IPV6与IPV4用户

    IPV6改造中经常会遇到,网站使用了CDN,但是CDN厂商还不支持IPV6的情况,而AAAA.A.CNAME记录互相冲突,想实现IPV6用户得到AAAA记录,IPV4用户得到CNAME记录的需求. 解 ...

  2. mkswap命令详解

    基础命令学习目录首页 原文链接:http://blog.51cto.com/arlen99/1743841 mkswap命令用于在一个文件或者设备上建立交换分区.在建立完之后要使用sawpon命令开始 ...

  3. leetcode 184 部门工资最高的员工

    题目描述:Employee 表包含所有员工信息,每个员工有其对应的 Id, salary 和 department Id. Department 表包含公司所有部门的信息. 编写一个 SQL 查询,找 ...

  4. Scrum Meeting 12 -2014.11.18

    今天的任务都比较顺利,测试暂时还没发现特别的问题. Member Today’s task Next task 林豪森 与其他小组商讨整合问题 与其他小组商讨整合问题 宋天舒 测试项目功能实现 测试项 ...

  5. 团队博客作业Week3 --- 项目选择&&需求疑问

    项目选择 经过团队内所有成员一致探讨,我们团队选择完善和改进之学霸系统的第二个子模块,即:网站内容结构定义和数据处理.具体的要求如下:(摘自Xueba系统项目需求) 网站内容结构定义和数据处理(Con ...

  6. java_web连接SQL_server详细步骤

    (1).我用的是Myeclipse,可以直接将sqljdbc4.jar拷到项目文件 (2).点开SQL Server配置管理器 选中SQL Server2008网络配置下的SQLEXPRESS的协议, ...

  7. Hibernate笔记③--集合映射、组合映射、联合主键、查询案例

    lazy 懒加载 默认为proxy   继承映射 discriminant column="type" type="string"   集合映射 生成表的语句: ...

  8. 先做一个用来测试的chrome浏览器插件

    如何制作chrome插件 在项目汇报中,有同学提到了想要了解如何制作插件,特写该篇博客供大家查阅~ 一个简单的插件需要manifest.json.popup.html.popup.js.content ...

  9. sqlserver trigger(触发器)-更新某几列数据时触发【转】

    CREATE TRIGGER [dbo].[updataAlarmLevel]ON [dbo].[Alarm_Alarm_Info]AFTER INSERT, UPDATE – 在更新和插入之后ASB ...

  10. QTime的本质上是一个int,QDateTime本质上是一个qint64

    研究这个问题的起因发现使用<=比较时间的不准确,所以怀疑是一个浮点数(Delphi里的time就是一个浮点数).结果却发现是一个int class Q_CORE_EXPORT QTime { e ...