HDU 3698 DP+线段树

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了HDU 3698 DP+线段树相关的知识,希望对你有一定的参考价值。

给出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+线段树的主要内容,如果未能解决你的问题,请参考以下文章

HDU 4521 小明系列问题——小明序列 (线段树维护DP)

[HDU4867]Xor (线段树分治+类数位dp)

hdu 6447(线段树+dp)

hdu 4747 线段树/DP

HDU 3607 线段树+DP+离散化

HDU4719-Oh My Holy FFF(DP线段树优化)