[UOJ430]line
首先有个暴力DP,设$s_i=\sum\limits_{j\geq i}w_j$,有$f_i=\min\limits_{l_i\lt j\leq i}f_{j-1}+s_{i+1}\max\{t_{j\cdots i}\}$
想用斜率优化,但是式中的max会随$i$改变而改变
考虑把单调栈的弹栈序列建成一棵树,每个点$x$存的直线为$y=t_xx+\min\{f_{fa_x+1\cdots x}\}$,这样一个点要查的直线就是到根路径的一条链,$l_i$的限制在最后一小段查$f$的区间最小值即可
这棵树有很好的性质,从根往下走的$t_x$是不升的,我们要查的$s_{i+1}$是递减的,所以可以树剖+线段树维护凸壳,边插入边维护即可
总时间复杂度$O(n\log^2n)$,感觉把单调栈建成树这一步非常厉害
#include<stdio.h>
#include<vector>
using namespace std;
typedef long long ll;
typedef double du;
const ll inf=1e18;
void fmin(ll&a,ll b){
if(b<a)a=b;
}
int l[100010],t[100010],w[100010],n;
ll s[100010];
int st[100010],tp;
int h[100010],nex[100010],to[100010],M;
void add(int a,int b){
M++;
to[M]=b;
nex[M]=h[a];
h[a]=M;
}
int fa[100010],siz[100010],son[100010];
void dfs(int x){
int i,k=-1;
siz[x]=1;
for(i=h[x];i;i=nex[i]){
fa[to[i]]=x;
dfs(to[i]);
siz[x]+=siz[to[i]];
if(k==-1||siz[to[i]]>siz[k])k=to[i];
}
son[x]=k;
}
int bl[100010],pos[100010],rp[100010];
void dfs(int x,int chain){
bl[x]=chain;
pos[x]=++M;
rp[M]=x;
if(~son[x])dfs(son[x],chain);
for(int i=h[x];i;i=nex[i]){
if(to[i]!=son[x])dfs(to[i],to[i]);
}
}
struct line{
ll k,b;
line(ll k=0,ll b=0):k(k),b(b){}
ll v(ll x){return k*x+b;}
};
du its(line a,line b){
return(b.b-a.b)/(du)(a.k-b.k);
}
typedef vector<line> vl;
vl g[400010];
#define ls g[g.size()-1]
void push(vl&g,line v){
if(!g.empty()&&ls.k==v.k){
if(ls.b<=v.b)return;
g.pop_back();
}
while(g.size()>1&&its(ls,g[g.size()-2])>=its(ls,v))g.pop_back();
g.push_back(v);
}
void modify(int p,line v,int l,int r,int x){
push(g[x],v);
if(l==r)return;
int mid=(l+r)>>1;
if(p<=mid)
modify(p,v,l,mid,x<<1);
else
modify(p,v,mid+1,r,x<<1|1);
}
ll get(vl&g,ll v){
while(g.size()>1&&its(ls,g[g.size()-2])>=v)g.pop_back();
return ls.v(v);
}
ll query(int L,int R,ll v,int l,int r,int x){
if(L<=l&&r<=R)return get(g[x],v);
int mid=(l+r)>>1;
ll res=inf;
if(L<=mid)fmin(res,query(L,R,v,l,mid,x<<1));
if(mid<R)fmin(res,query(L,R,v,mid+1,r,x<<1|1));
return res;
}
ll f[100010];
ll T[400010];
int N;
void build(){
for(N=1;N<=n+1;N<<=1);
for(int i=1;i<N<<1;i++)T[i]=inf;
}
void modify(int x,ll v){
T[x+=N]=v;
for(x>>=1;x;x>>=1)T[x]=min(T[x<<1],T[x<<1|1]);
}
ll query(int s,int t){
ll res=inf;
for(s+=N-1,t+=N+1;s^t^1;s>>=1,t>>=1){
if(~s&1)fmin(res,T[s^1]);
if(t&1)fmin(res,T[t^1]);
}
return res;
}
void work(int x){
int u,l,r,mid;
ll res;
modify(x,f[x-1]);
modify(pos[x],line(t[x],query(fa[x]+1,x)),1,M,1);
res=inf;
for(u=x;;){
if(::l[x]>fa[bl[u]]){
l=pos[bl[u]];
r=pos[u];
while(l<r){
mid=(l+r)>>1;
if(rp[mid]>=::l[x])
r=mid;
else
l=mid+1;
}
if(pos[u]>l)fmin(res,query(l+1,pos[u],s[x+1],1,M,1));
u=rp[l];
break;
}else{
fmin(res,query(pos[bl[u]],pos[u],s[x+1],1,M,1));
u=fa[bl[u]];
}
}
fmin(res,query(::l[x],u)+t[u]*s[x+1]);
f[x]=res;
}
int main(){
int i;
scanf("%d",&n);
for(i=1;i<=n;i++){
scanf("%d%d%d",l+i,t+i,w+i);
l[i]++;
}
for(i=n;i>0;i--)s[i]=s[i+1]+w[i];
for(i=1;i<=n;i++){
while(tp&&t[i]>t[st[tp]]){
add(st[tp-1],st[tp]);
tp--;
}
st[++tp]=i;
}
for(i=1;i<=tp;i++)add(st[i-1],st[i]);
dfs(0);
M=0;
dfs(0,0);
fa[0]=-1;
build();
for(i=1;i<=n;i++)work(i);
printf("%lld",f[n]);
}
[UOJ430]line的更多相关文章
- ILJMALL project过程中遇到Fragment嵌套问题:IllegalArgumentException: Binary XML file line #23: Duplicate id
出现场景:当点击"分类"再返回"首页"时,发生error退出 BUG描述:Caused by: java.lang.IllegalArgumentExcep ...
- Error on line -1 of document : Premature end of file. Nested exception: Premature end of file.
启动tomcat, 出现, ( 之前都是好好的... ) [lk ] ERROR [08-12 15:10:02] [main] org.springframework.web.context.Con ...
- 关于xml加载提示: Error on line 1 of document : 前言中不允许有内容
我是在java中做的相关测试, 首先粘贴下报错: 读取xml配置文件:xmls\property.xml org.dom4j.DocumentException: Error on line 1 of ...
- Eclipse "Unable to install breakpoint due to missing line number attributes..."
Eclipse 无法找到 该 断点,原因是编译时,字节码改变了,导致eclipse无法读取对应的行了 1.ANT编译的class Eclipse不认,因为eclipse也会编译class.怎么让它们统 ...
- Linix登录报"/etc/profile: line 11: syntax error near unexpected token `$'{\r''"
同事反馈他在一测试服务器(CentOS Linux release 7.2.1511)上修改了/etc/profile文件后,使用source命令不能生效,让我帮忙看看,结果使用SecureCRT一登 ...
- [LeetCode] Line Reflection 直线对称
Given n points on a 2D plane, find if there is such a line parallel to y-axis that reflect the given ...
- [LeetCode] Tenth Line 第十行
How would you print just the 10th line of a file? For example, assume that file.txt has the followin ...
- [LeetCode] Max Points on a Line 共线点个数
Given n points on a 2D plane, find the maximum number of points that lie on the same straight line. ...
- "Installation failed !" in GUI but not in CLI (/usr/bin/winusb: line 78: 18265 Terminated )
"Installation failed !" in GUI but not in CLI (/usr/bin/winusb: line 78: 18265 Terminated ...
随机推荐
- 利用thrift rpc进行C++与Go的通信
一:什么是rpc rpc通俗来理解就是远程调用函数,相对于本地调用来说,只需要在主调函数中调用被掉函数即可,代码如下: void fun(int i) { cout << "fu ...
- Java编程思想 4th 第1章 对象导论
所有编程语言都提供抽象机制. 面向对象编程似乎是一种很好的编程思想和方式,面向对象编程中的对象简洁描述是:对象具有状态.行为和标识.状态指的是数据存储,存储的数据能反应状态:行为指的是方法,方法表示对 ...
- 【多视图几何】TUM 课程 第1章 数学基础:线性代数
在 YouTube 上找到了慕尼黑工业大学(Technische Universitaet München)计算机视觉组 Daniel Cremers 教授的 Multiple View Geomet ...
- 【算法学习】manacher
manacher太水了. 这篇blog不能称作算法学习,因为根本没有介绍…… 就贴个模板,太简单了…… #include<cstdio> #include<cstring> # ...
- ajax代码示例
function loadXMLDoc(idName,url,sendOut) { var xmlhttp; if (window.XMLHttpRequest) {// code for IE7+, ...
- [转载]Windows服务编写原理及探讨(4)
(四)一些问题的讨论 前面几章的内容都是服务的一些通用的编写原理,但里面隐含着一些问题,编写简单的服务时看不出来,但遇到复杂的应用就会出现一些问题,所以本章就是用来分析.解决这些问题的,适用于高级应用 ...
- 10 The Go Programming Language Specification go语言规范 重点
The Go Programming Language Specification go语言规范 Version of May 9, 2018 Introduction 介绍 Notation 符号 ...
- Java线程的阻塞
线程的阻塞 线程的优先级 线程总是存在优先级,优先级范围在1~10之间,线程默认优先级是5(数值越大优先级越高): JVM线程调度程序是基于优先级的抢先调度机制: 在大多数情况下,当前运行的线程优先级 ...
- find命令的使用
在以.conf结尾的文件里面查找含有aaa字符串的那一行 ( -name后面可以写 "*.*" 即匹配所有的文件 ) find / -name "*.conf& ...
- Spring框架的基本使用(AOP部分)
AOP,Aspect Oriented Programming,意为面向切面编程,是通过预编译方式和运行期间动态代理实现程序功能的统一维护的一种技术.AOP采取横向抽取机制,取代了传统纵向继承体系重复 ...