可以很容易的写出dp方程:

F[i][j]=min(F[l][j-1]+w[l][i])+c[i] (w[i][j]是从l+1到i-1这些点p里,所有满足d[p]+s[p]<d[i] && d[p]-s[p]>d[l]的点的w[p]之和)

考虑i+1,会对方程造成什么变化

w[l][i]会变得跟大了(满足d[p]+s[p]<d[i]的点更多了)

可以发现增长的是一个区间,求和也是一个区间

线段树解决

CODE:

#include<cstdio>

#include<iostream>

#include<cstring>

#include<algorithm>

#include<vector>

using namespace std;

#define maxn 20100

#define inf 1000000000

struct node {

int l,r,mi,lz;

}t[maxn*8];

#define lc(x) (x<<1)

#define rc(x) ((x<<1)+1)

#define mi(x) t[x].mi

#define lz(x) t[x].lz

#define mid ((l+r)>>1)

#define update(x) mi(x)=min(mi(lc(x)),mi(rc(x)))

int f[maxn],pf[maxn];

int build(int x,int l,int r) {

t[x].l=l,t[x].r=r;

t[x].lz=0;

if (l==r) return t[x].mi=pf[l];

build(lc(x),l,mid);

build(rc(x),mid+1,r);

update(x);

return 0;

}

int pushback(int x) {

if (t[x].l==t[x].r) return 0;

lz(lc(x))+=lz(x),lz(rc(x))+=lz(x);

mi(lc(x))+=lz(x),mi(rc(x))+=lz(x);

lz(x)=0;

}

int query(int x,int x1,int y1){

int l=t[x].l,r=t[x].r;

if (y1<l||x1>r) return inf;

if (x1<=l&&y1<=r) return mi(x);

pushback(x);

return min(query(lc(x),x1,y1),query(rc(x),x1,y1));

}

int add(int x,int x1,int y1,int z) {

int l=t[x].l,r=t[x].r;

if (y1<l||x1>r) return 0;

if (x1<=l&&y1>=r) return lz(x)+=z,mi(x)+=z;

pushback(x);

add(lc(x),x1,y1,z),add(rc(x),x1,y1,z);

update(x);

}

int n,k;

int c[maxn],s[maxn],d[maxn],w[maxn];

int st[maxn],ed[maxn];

vector<int> list[maxn];

int solve(){

int sum=0;

for (int i=1;i<=n;i++) {

f[i]=sum+c[i];

for(int j=0;j<list[i].size();j++) sum+=w[list[i][j]];

}

int ans=f[n];

for (int i=1;i<=k;i++) {

//for (int i=1;i<=n;i++) printf("%d ",f[i]);printf("\n");

memcpy(pf,f,sizeof(pf));

for (int j=1;j<=n;j++) f[j]=inf;

build(1,1,n);

for (int j=1;j<=n;j++){

f[j]=min(f[j],query(1,1,j-1))+c[j];

for (int k=0;k<list[j].size();k++) add(1,1,st[list[j][k]]-1,w[list[j][k]]);

}

ans=min(ans,f[n]);

}

printf("%d\n",ans);

return 0;

}

int main(){

scanf("%d%d",&n,&k);

for (int i=2;i<=n;i++) scanf("%d",d+i);

for (int i=1;i<=n;i++) scanf("%d",c+i);

for (int i=1;i<=n;i++) scanf("%d",s+i);

for (int i=1;i<=n;i++) scanf("%d",w+i);

d[++n]=inf,c[n]=w[n]=s[n]=0;

for (int i=1;i<=n;i++) {

st[i]=lower_bound(d+1,d+1+n,d[i]-s[i])-d;

ed[i]=upper_bound(d+1,d+1+n,d[i]+s[i])-d-1;

list[ed[i]].push_back(i);

}

solve();

return 0;

}

BZOJ 1835: [ZJOI2010]base 基站选址(DP,线段树)的更多相关文章

  1. BZOJ 1835 [ZJOI2010]base 基站选址:线段树优化dp

    传送门 题意 有 $ n $ 个村庄在一排直线上,现在要建造不超过 $ K $ 个通讯基站,基站只能造在村庄处. 第 $ i $ 个村庄距离第 $ 1 $ 个村庄的距离为 $ D_i $ .在此建造基 ...

  2. BZOJ 1835: [ZJOI2010]base 基站选址 [序列DP 线段树]

    1835: [ZJOI2010]base 基站选址 题目描述 有N个村庄坐落在一条直线上,第i(i>1)个村庄距离第1个村庄的距离为Di.需要在这些村庄中建立不超过K个通讯基站,在第i个村庄建立 ...

  3. bzoj 1835 [ZJOI2010]base 基站选址(DP+线段树)

    [题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=1835 [题意] 有n个村庄,每个村庄位于d[i],要求建立不多于k个基站,在第i个村庄 ...

  4. BZOJ1835: [ZJOI2010]base 基站选址【线段树优化DP】

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

  5. 2018.11.06 bzoj1835: [ZJOI2010]base 基站选址(线段树优化dp)

    传送门 二分出每个点不需要付www贡献的范围,然后可以推出转移式子: f[i][j]=f[i−1][k]+value(k+1,j)+c[i]f[i][j]=f[i-1][k]+value(k+1,j) ...

  6. bzoj 1835: [ZJOI2010]base 基站选址

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

  7. bzoj[1835][ZJOI2010]base 基地选址

    bzoj[1835][ZJOI2010]base 基地选址 标签: 线段树 DP 题目链接 题解 这个暴力DP的话应该很容易看出来. dp[i][j]表示造了i个通讯站,并且j是第i个的最小费用. \ ...

  8. bzoj 1835 base 基站选址 - 动态规划 - 线段树

    题目传送门 需要高级权限的传送门 题目大意 有$n$个村庄坐落在一条直线上,第$i \ \ \ (i>1)$个村庄距离第$1$个村庄的距离为$D_i$.需要在这些村庄中建立不超过$K$个通讯基站 ...

  9. BZOJ 1835 基站选址(DP+线段树)

    # include <cstdio> # include <cstring> # include <cstdlib> # include <iostream& ...

随机推荐

  1. bzoj-4318 OSU! 【数学期望】

    Description osu 是一款群众喜闻乐见的休闲软件.  我们可以把osu的规则简化与改编成以下的样子:  一共有n次操作,每次操作只有成功与失败之分,成功对应1,失败对应0,n次操作对应为1 ...

  2. C语言-指针、数组、结构体、分支、循环混合使用

    1.写一个程序,输出如下内容: //############################################################# //### name number ma ...

  3. python中使用urllib2伪造HTTP报头的2个方法

    在采集网页信息的时候,经常需要伪造报头来实现采集脚本的有效执行 下面,我们将使用urllib2的header部分伪造报头来实现采集信息 方法1. ? 1 2 3 4 5 6 7 8 9 10 11 1 ...

  4. YII 1.0 发表文章用到的小物件

    <?php $form = $this->beginWidget('CActiveForm',array('htmlOptions'=>array('enctype'=>'mu ...

  5. zepto.js 处理Touch事件

    处 理Touch事件能让你了解到用户的每一根手指的位置,在touch事件触发的时候产生,可以通过touch event handler的event对象取到,如果基于zepto.js开发,一般是通过ev ...

  6. thinkPHP 模板中的语法

    一.导入CSS和JS文件   1.css link       js  scr        <link rel='stylesheet' type='text/css' href='__PUB ...

  7. 【转】 如何提高自己的acm个人能力

    2011-05-17 21:26 1429人阅读 评论(0) 收藏 举报 算法网络blog八卦游戏读书 转载自 简单de数字 最终编辑 fading_code by   zfy0701 本来以为HNU ...

  8. redhat+11g+rac 安装数据库软件时只有一个节点可选

    在安装数据库软件时,只能检测到一个节点 650) this.width=650;" title="捕获.JPG" src="http://s3.51cto.co ...

  9. HTML 5 Web 存储、应用程序缓存、Web Workers

    在客户端存储数据 HTML5 提供了两种在客户端存储数据的新方法: localStorage - 没有时间限制的数据存储 sessionStorage - 针对一个 session 的数据存储 之前, ...

  10. 使用IDEA开发及测试Spark的环境搭建及简单测试

    一.安装JDK(具体安装省略) 二.安装Scala(具体安装省略) 三.安装IDEA 1.打开后会看到如下,然后点击OK