给出N*M矩阵。每一个点建立灯塔有花费。每一个点的灯塔有连接范围,求每一行都建立一个灯塔的最小花费,要求每相邻两行的灯塔能够互相连接。满足 |j-k|≤f(i,j)+f(i+1,k)

DP思路,dp[i][j]=在第i行的j位置放置灯塔的最小花费。dp[i][j]=Min(dp[i-1][k]+a[i][j]),同一时候更新当前点能够覆盖该行的全部位置的最小值

要求上个区间的最小值,用线段树优化,否则超时

滚动数组,否则超内存

#include "stdio.h"
#include "string.h" int inf=0x3f3f3f3f; struct node
{
int Max,Min,lazy; // min记录区间最小值。max记录区间的最小值的最大值
short int l,r;
}data[2][20100]; int a[110][5010],b[110][5010]; int Min(int a,int b)
{
if (a<b) return a;
else return b;
} void Pushdown(int w,int k)
{
if (data[w][k].l==data[w][k].r) return ;
if (data[w][k].lazy==-1) return ; if (data[w][k].lazy<data[w][k*2].Min)
data[w][k*2].Min=data[w][k*2].lazy=data[w][k].lazy; if (data[w][k].lazy<data[w][k*2+1].Min)
data[w][k*2+1].Min=data[w][k*2+1].lazy=data[w][k].lazy; data[w][k].lazy=-1;
} void build(int w,int l,int r,int k)
{
int mid;
data[w][k].l=l;
data[w][k].r=r;
data[w][k].lazy=-1;
data[w][k].Min=inf;
data[w][k].Max=inf; if (l==r) return ; mid=(l+r)/2; build(w,l,mid,k*2);
build(w,mid+1,r,k*2+1);
} void updata(int w,int l,int r,int k,int op)
{
int mid; if (data[w][k].l==data[w][k].r)
{
if (data[w][k].Min>op) data[w][k].Max=data[w][k].Min=op;
return ;
}
if (data[w][k].l==l && data[w][k].r==r )
{
mid=(data[w][k].l+data[w][k].r)/2; if (op<data[w][k].Max) // 若以下区间存在最小值>op,则继续向下更新
{
updata(w,l,mid,k*2,op);
updata(w,mid+1,r,k*2+1,op);
data[w][k].Max=op;
}
if (op<data[w][k].Min)
data[w][k].Min=data[w][k].lazy=data[w][k].Max=op; return ;
} Pushdown(w,k); mid=(data[w][k].l+data[w][k].r)/2; if (r<=mid) updata(w,l,r,k*2,op);
else
if (l>mid) updata(w,l,r,k*2+1,op);
else
{
updata(w,l,mid,k*2,op);
updata(w,mid+1,r,k*2+1,op);
} data[w][k].Min=Min(data[w][k*2].Min,data[w][k*2+1].Min); } int query(int w,int l,int r,int k)
{
int mid;
if (data[w][k].l==l && data[w][k].r==r)
return data[w][k].Min; Pushdown(w,k); mid=(data[w][k].l+data[w][k].r)/2; if (r<=mid) return query(w,l,r,k*2);
else
if (l>mid) return query(w,l,r,k*2+1);
else
return Min(query(w,l,mid,k*2),query(w,mid+1,r,k*2+1));
} void pri(int w,int k)
{
if (data[w][k].l==data[w][k].r)
{
printf("%d ",data[w][k].Min);
return ;
}
Pushdown(w,k); pri(w,k*2);
pri(w,k*2+1);
}
int main()
{
int n,m,i,j,l,r,x;
while (scanf("%d%d",&n,&m)!=EOF)
{
if (n==0 && m==0) break;
for (i=1;i<=n;i++)
for (j=1;j<=m;j++)
scanf("%d",&a[i][j]); for (i=1;i<=n;i++)
for (j=1;j<=m;j++)
scanf("%d",&b[i][j]); build(1,1,m,1);
for (i=1;i<=m;i++)
{
l=i-b[1][i]; if (l<1) l=1;
r=i+b[1][i]; if (r>m) r=m;
updata(1,l,r,1,a[1][i]); // 初始化第一行每个位置的最优值
} for (i=2;i<=n;i++)
{
build(i%2,1,m,1);
for (j=1;j<=m;j++)
{
l=j-b[i][j]; if (l<1) l=1;
r=j+b[i][j]; if (r>m) r=m;
x=query(1-i%2,l,r,1); // 对于每一行的每个位置。查找上一行能够连接到该位置的最小值
updata(i%2,l,r,1,x+a[i][j]); // 更新该行该位置能够覆盖到的全部位置的最小值
}
} /* for (i=1;i<=n;i++)
{
printf("\n\n");
pri(i,1); }
printf("\n");*/ //debug printf("%d\n",data[n%2][1].Min); }
return 0;
}

HDU 3698 DP+线段树的更多相关文章

  1. hdu 3016 dp+线段树

    Man Down Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total S ...

  2. ZOJ 3349 Special Subsequence 简单DP + 线段树

    同 HDU 2836 只不过改成了求最长子串. DP+线段树单点修改+区间查最值. #include <cstdio> #include <cstring> #include ...

  3. hdu 4031 attack 线段树区间更新

    Attack Time Limit: 5000/3000 MS (Java/Others)    Memory Limit: 65768/65768 K (Java/Others)Total Subm ...

  4. hdu 4288 离线线段树+间隔求和

    Coder Time Limit: 20000/10000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others) Total Su ...

  5. cf834D(dp+线段树区间最值,区间更新)

    题目链接: http://codeforces.com/contest/834/problem/D 题意: 每个数字代表一种颜色, 一个区间的美丽度为其中颜色的种数, 给出一个有 n 个元素的数组, ...

  6. Codeforces Round #620 F2. Animal Observation (hard version) (dp + 线段树)

    Codeforces Round #620 F2. Animal Observation (hard version) (dp + 线段树) 题目链接 题意 给定一个nm的矩阵,每行取2k的矩阵,求总 ...

  7. 题解 HDU 3698 Let the light guide us Dp + 线段树优化

    http://acm.hdu.edu.cn/showproblem.php?pid=3698 Let the light guide us Time Limit: 5000/2000 MS (Java ...

  8. HDU 3698 Let the light guide us(DP+线段树)(2010 Asia Fuzhou Regional Contest)

    Description Plain of despair was once an ancient battlefield where those brave spirits had rested in ...

  9. HDU 4719 Oh My Holy FFF(DP+线段树)(2013 ACM/ICPC Asia Regional Online ―― Warmup2)

    Description N soldiers from the famous "*FFF* army" is standing in a line, from left to ri ...

随机推荐

  1. 【C++第二课】---C到C++的函数升级

    C++中对C语言在函数使用方面做了很大的升级 一﹑内联函数 1.C++中推荐使用内联函数来替代宏片段代码 2.C++中使用关键字inline声明内联函数 例如: inline int func(int ...

  2. HDU 1796 Howmany integers can you find (容斥原理)

    How many integers can you find Time Limit: 12000/5000 MS (Java/Others)    Memory Limit: 65536/32768 ...

  3. 我使用过的Linux命令之file - 检测并显示文件类型

    摘自:http://codingstandards.iteye.com/blog/804463 我使用过的Linux命令之file - 检测并显示文件类型 用途说明 file命令是用来检测并显示文件类 ...

  4. Swift自定义Class实现Hashable

    假如有个Bit类,其中含有CGPoint类型的point属性,Class定义如下 class Bit { var point : CGPoint init(point : CGPoint) { sel ...

  5. 数字温湿度传感器DHT11--操作源代码

    //IO定义 #define P_DataIN_DHT11 PB0_IN #define P_DataOUT_DHT11 PB0_OUT //宏定义 #define BSET_DHT11 P_Data ...

  6. 引用枚举进行对比时 enum需强制转换

    枚举类 public enum MailRead { /// <summary> /// 未读 /// </summary> UNREAD=0, /// <summary ...

  7. C#中按指定质量保存图片的实例代码 24位深度

     /// <summary>        /// 按指定的压缩质量及格式保存图片(微软的Image.Save方法保存到图片压缩质量为75)        /// </summary ...

  8. UVA 1001 Say Cheese

    题意: 一只母老鼠想要找到她的玩具,而玩具就丢在一个广阔的3维空间上某个点,而母老鼠在另一个点,她可以直接走到达玩具的位置,但是耗时是所走过的欧几里得距离*10s.还有一种方法,就是靠钻洞,洞是球形的 ...

  9. jQuery基础---Ajax基础教程(二)

    jQuery基础---Ajax进阶 内容提纲: 1.加载请求 2.错误处理 3.请求全局事件 4.JSON 和 JSONP 5.jqXHR 对象 发文不易,转载请注明出处! 在 Ajax 基础一篇中, ...

  10. (八)Android广播接收器BroadcastReceiver

    一.使用Broadcast Reciver 1.右击java文件夹,new->other->Broadcast Receiver后会在AndroidManifest.xml文件中生成一个r ...