原题传送门:P2604 [ZJOI2010]基站选址

看一眼题目,变知道这题一定是dp

设f[i][j]表示在第i个村庄修建第j个基站且不考虑i+1~n个村庄的最小费用

可以得出f[i][j] = Min(f[k][j - 1] + cost[k][i] ) + c[i] (j - 1 <= k < i)

其中cost[k][i]表示i~k之间没有基站所需要的费用

计算复杂度O(N),加上循环,总复杂度O(N^2 K)

看一下数据范围K <= N,K <= 100 , N <= 20000

真棒,TLE

考虑如何优化

首先,我们发现之前的转移方程可以去掉一维j,实际上只要在最外层枚举j就可以了

f[i] = Min( f[k] + cost[k][i] ) + c[i] (j - 1 <= k < i)

主要的时间浪费在计算cost上

我们要找方法来优化

对于任意一个村庄i,记它所能被覆盖的左右边界st[i],ed[i](最左端、最右端可以覆盖到i的基站位置,可用二分查找处理)

然后在用邻接表记录ed值为i的村庄有哪些,在这些村庄之前建立基站就覆盖不到i了。

这样当我们推导i + 1时,若从村庄1~st[k] - 1(ed[k] = i)转移过来则必定要赔偿村庄k的费用,我们就可以考虑用线段树来维护f[k] + cost[k][i]的值

即在区间[1, st[k] - 1]加上村庄k的费用,而转移即在区间[1, i - 1]找f[k] + cost[k][i]的最小值,总复杂度为O(N log N K)。

#pragma GCC optimize("O3")
#include <bits/stdc++.h>
#define N 20005
#define inf 0x3f3f3f3f
using namespace std;
inline int read()
{
register int x=0,f=1;register char ch=getchar();
while(ch<'0'||ch>'9'){if(ch=='-')f=-1;ch=getchar();}
while(ch>='0'&&ch<='9'){x=x*10+ch-'0';ch=getchar();}
return x*f;
}
inline int Min(register int a,register int b)
{
return a<b?a:b;
}
int D[N],C[N],S[N],W[N],lr[N],rr[N],f[N];
vector<int> vq[N];
int ql,qr;
struct Segment_tree{
int minn[N<<2],tag[N<<2];
inline void pushup(register int x)
{
minn[x]=Min(minn[x<<1],minn[x<<1|1]);
}
inline void build(register int x,register int l,register int r)
{
tag[x]=0;
if(l==r)
{
minn[x]=f[l];
return;
}
int mid=(l+r)>>1;
build(x<<1,l,mid);
build(x<<1|1,mid+1,r);
pushup(x);
}
inline void maintain(register int x,register int l,register int r)
{
if(l!=r)
pushup(x);
minn[x]+=tag[x];
}
inline void update(register int x,register int l,register int r,register int v)
{
if(ql<=l&&qr>=r)
tag[x]+=v;
else
{
int mid=(l+r)>>1;
if(ql<=mid)
update(x<<1,l,mid,v);
if(qr>mid)
update(x<<1|1,mid+1,r,v);
}
maintain(x,l,r);
}
inline int query(register int x,register int l,register int r,register int v)
{
if(ql<=l&&qr>=r)
return minn[x]+v;
int ret=inf,mid=(l+r)>>1;
if(ql<=mid)
ret=Min(ret,query(x<<1,l,mid,v+tag[x]));
if(qr>mid)
ret=Min(ret,query(x<<1|1,mid+1,r,v+tag[x]));
return ret;
}
}T;
int main()
{
int n=read(),k=read();
for(register int i=2;i<=n;++i)
D[i]=read();
for(register int i=1;i<=n;++i)
C[i]=read();
for(register int i=1;i<=n;++i)
S[i]=read();
for(register int i=1;i<=n;++i)
W[i]=read();
++n,++k;
D[n]=inf;
for(register int i=1;i<=n;++i)
{
lr[i]=lower_bound(D+1,D+n+1,D[i]-S[i])-D;
rr[i]=lower_bound(D+1,D+n+1,D[i]+S[i])-D;
if(D[i]+S[i]<D[rr[i]])
--rr[i];
vq[rr[i]].push_back(i);
}
int ans=inf;
for(register int j=1;j<=k;++j)
{
if(j==1)
{
int tot=0;
for(register int i=1;i<=n;++i)
{
f[i]=tot+C[i];
for(register int tmp=0;tmp<vq[i].size();++tmp)
tot+=W[vq[i][tmp]];
}
ans=Min(ans,f[n]);
continue;
}
T.build(1,1,n);
for(register int i=1;i<=n;++i)
{
ql=1,qr=i-1;
int add=qr?T.query(1,1,n,0):0;
f[i]=add+C[i];
for(register int tmp=0;tmp<vq[i].size();++tmp)
{
ql=1,qr=lr[vq[i][tmp]]-1;
if(qr>0)
T.update(1,1,n,W[vq[i][tmp]]);
}
}
ans=Min(ans,f[n]);
}
printf("%d",ans);
return 0;
}

【题解】Luogu P2605 [ZJOI2010]基站选址的更多相关文章

  1. luogu P2605 [ZJOI2010]基站选址 线段树优化dp

    LINK:基站选址 md气死我了l达成1结果一直调 显然一个点只建立一个基站 然后可以从左到右进行dp. \(f_{i,j}\)表示强制在i处建立第j个基站的最小值. 暴力枚举转移 复杂度\(n\cd ...

  2. luogu P2605 [ZJOI2010]基站选址

    luogu 先考虑朴素dp,设\(f_{i,j}\)表示在第\(i\)个村庄放了基站,一共放了\(j\)次,且只考虑前面村庄影响的答案.这里可以把\(j\)放在外面枚举,然后从\(f_{k,j-1}( ...

  3. P2605 [ZJOI2010]基站选址

    题目描述 有N个村庄坐落在一条直线上,第i(i>1)个村庄距离第1个村庄的距离为Di.需要在这些村庄中建立不超过K个通讯基站,在第i个村庄建立基站的费用为Ci.如果在距离第i个村庄不超过Si的范 ...

  4. 洛谷$P2605\ [ZJOI2010]$基站选址 线段树优化$dp$

    正解:线段树优化$dp$ 解题报告: 传送门$QwQ$ 难受阿,,,本来想做考试题的,我还造了个精妙无比的题面,然后今天讲$dp$的时候被讲到了$kk$ 先考虑暴力$dp$?就设$f_{i,j}$表示 ...

  5. 题解 [ZJOI2010]基站选址

    题解 [ZJOI2010]基站选址 题面 解析 首先考虑一个暴力的DP, 设\(f[i][k]\)表示第\(k\)个基站设在第\(i\)个村庄,且不考虑后面的村庄的最小费用. 那么有\(f[i][k] ...

  6. 【LG2605】[ZJOI2010]基站选址

    [LG2605][ZJOI2010]基站选址 题面 洛谷 题解 先考虑一下暴力怎么写,设\(f_{i,j}\)表示当前\(dp\)到\(i\),且强制选\(i\),目前共放置\(j\)个的方案数. 那 ...

  7. [ZJOI2010]基站选址,线段树优化DP

    G. base 基站选址 内存限制:128 MiB 时间限制:2000 ms 标准输入输出 题目类型:传统 评测方式:文本比较   题目描述 有N个村庄坐落在一条直线上,第i(i>1)个村庄距离 ...

  8. bzoj 1835/luogu P2605 : [ZJOI2010]base 基站选址

    题目描述 有N个村庄坐落在一条直线上,第i(i>1)个村庄距离第1个村庄的距离为Di.需要在这些村庄中建立不超过K个通讯基站,在第i个村庄建立基站的费用为Ci.如果在距离第i个村庄不超过Si的范 ...

  9. 题解 P2605 【[ZJOI2010]基站选址】(From luoguBlog)

    线段树优化dp 数组f[i][j]表示在前i个村庄内,第j个基站建在i处的最小费用 根据交线牛逼法和王鹤松式可得方程 f[i][j]=min(f[k][j−1]+cost(k,i)) cost(k,i ...

随机推荐

  1. iOS UI进阶-1.0 Quartz2D

    概述 Quartz 2D是一个二维绘图引擎,同时支持iOS和Mac系统.Quartz 2D能完成的工作: 绘制图形 : 线条\三角形\矩形\圆\弧等 绘制文字 绘制\生成图片(图像) 读取\生成PDF ...

  2. Windows Server 2008 安装 10.2.0.5 单实例

    需求:Windows Server 2008 安装 10.2.0.5 单实例 原以为非常简单的一次任务,实际却遇到了问题,故记录一下. 1.安装10.2.0.1 2.安装10.2.0.4 3.安装10 ...

  3. Spring+SpringMVC+MyBatis整合应用

    1)搭建Spring,SpringMVC和MyBatis环境 创建一个web工程 添加MyBatis相关环境 引入数据库驱动包和DBCP连接池开发包 引入MyBatis开发包 添加Spring,Spr ...

  4. 35.HTML--网页自动跳转 5种方法

    网页自动跳转 5种方法 利用HTML标记(META中的REFRESH属性) 用HTML标记,就是用META的REFRESH标记,举例如下: <meta http-equiv=refresh co ...

  5. Flume:sink.type=hive

    Flume以Kafka为Source,以Hive为Sink进行数据转存. 业务背景:公司要求将某四川城市的卡口数据实时接入大数据平台中,历史数据可以通过Hive进行Load,也就是增量数据的对接问题. ...

  6. LA 3890 Most Distant Point from the Sea(半平面交)

    Most Distant Point from the Sea [题目链接]Most Distant Point from the Sea [题目类型]半平面交 &题解: 蓝书279 二分答案 ...

  7. whu 643 Soul Artist(二维BIT 区间更新,单点查询)

    Soul Artis [题目链接]Soul Artis [题目类型]二维BIT &题解: 二维区间更新和一维相比,要容斥一下,更新一块区间就是更新4个点. 还有这个我先是写了2*n^2logn ...

  8. 关于hibernate一级缓冲和二级缓冲

    关于一级缓冲和二级缓冲的内容,在面试的时候被问起来了,回答的不是很满意,所以有专门找了些有关这方面的文章加以理解 出自:http://blog.csdn.net/zdp072/article/deta ...

  9. Yii2开发小技巧

    工作中或多或少会用到的关于 Yii2 的小技巧的一个总结,包括model.controller.view或者配置文件的一些写法. 模型相关 获取查询SQL $query = User::find()- ...

  10. EasyUI表格DataGrid前端分页和后端分页的总结

    Demo简介 Demo使用Java.Servlet为后台代码(数据库已添加数据),前端使用EasyUI框架,后台直接返回JSON数据给页面 1.配置Web.xml文件 <?xml version ...