星星之火

[P3097] [USACO13DEC] [BZOJ4094] 最优挤奶Optimal Milking 解题报告(线段树+DP)

 题目链接:https://www.luogu.org/problemnew/show/P3097#sub

题目描述

Farmer John has recently purchased a new barn containing N milking machines (1 <= N <= 40,000), conveniently numbered 1..N and arranged in a row.

Milking machine i is capable of extracting M(i) units of milk per day (1 <= M(i) <= 100,000). Unfortunately, the machines were installed so close together that if a machine i is in use on a particular day, its two neighboring machines cannot be used that day (endpoint machines have only one neighbor, of course). Farmer John is free to select different subsets of machines to operate on different days.

Farmer John is interested in computing the maximum amount of milk he can extract over a series of D days (1 <= D <= 50,000). At the beginning of each day, he has enough time to perform maintenance on one selected milking machine i, thereby changing its daily milk output M(i) from that day forward. Given a list of these daily modifications, please tell Farmer John how much milk he can produce over D days (note that this number might not fit into a 32-bit integer).

FJ最近买了1个新仓库, 内含N 个挤奶机,1 到N 编号并排成一行。

挤奶机i 每天能产出M(i) 单位的奶。不幸的是, 机器装得太近以至于如果一台机器i 在某天被使用, 那与它相邻的两台机器那一天不能被使用

(当然, 两端点处的机器分别只有一个与之相邻的机器)。

FJ 可自由选择不同的机器在不同的日子工作。

FJ感兴趣于计算在D 天内他能产出奶的最大值。在每天开始时, 他有足够的时间维护一个选中的挤奶机i, 从而改变它从那天起的每日产奶量M(i)。

给出这些每日的修改,请告诉FJ他D 天中能产多少奶。

输入输出格式

输入格式:

* Line 1: The values of N and D.

* Lines 2..1+N: Line i+1 contains the initial value of M(i).

* Lines 2+N..1+N+D: Line 1+N+d contains two integers i and m,

indicating that Farmer John updates the value of M(i) to m at the beginning of day d.

输出格式:

* Line 1: The maximum total amount of milk FJ can produce over D days.

输入输出样例

输入样例#1: 
5 3 
1 
2 
3 
4 
5 
5 2 
2 7 
1 10 
输出样例#1: 
32 

说明

There are 5 machines, with initial milk outputs 1,2,3,4,5. On day 1, machine 5 is updated to output 2 unit of milk, and so on.

On day one, the optimal amount of milk is 2+4 = 6 (also achievable as 1+3+2). On day two, the optimal amount is 7+4 = 11. On day three, the optimal amount is 10+3+2=15.

题意简述:给定n个点排成一排,每个点有一个点权,多次改变某个点的点权并将最大点独立集计入答案,输出最终的答案

每次修改就是线段树单点修改操作,只需要对每个节点维护f数组就好了

#include<iostream>
#include<cstdio>
#include<algorithm>
#define ll long long
using namespace std;

const int maxn=4e4+15;
int n,d;
ll ans;
int a[maxn];
struct Tree
{
    int l,r;
    ll f[2][2];
}t[maxn<<2];
inline int read()
{
    char ch=getchar();
    int s=0,f=1;
    while (!(ch>='0'&&ch<='9')){if (ch=='-') f=-1;ch=getchar();}
    while (ch>='0'&&ch<='9') {s=(s<<3)+(s<<1)+ch-'0';ch=getchar();}
    return s*f;
}
void update(int rt)
{
    for (int i=0;i<=1;i++)
        for (int j=0;j<=1;j++)
        {
            ll p=max(t[rt<<1].f[i][0]+t[rt<<1|1].f[0][j],t[rt<<1].f[i][1]+t[rt<<1|1].f[0][j]);
            p=max(p,t[rt<<1].f[i][0]+t[rt<<1|1].f[1][j]);
            t[rt].f[i][j]=p;
        }
}
void build(int rt,int l,int r)
{
    t[rt].l=l;t[rt].r=r;
    if (l==r)
    {
        t[rt].f[1][1]=1ll*a[l];
        return;
    }
    int mid=l+r>>1;
    build(rt<<1,l,mid);
    build(rt<<1|1,mid+1,r);
    update(rt);
}
void change(int rt,int x,int v)
{
    if (t[rt].l==t[rt].r)
    {
        t[rt].f[1][1]=1ll*v;
        return;
    }
    int mid=t[rt].l+t[rt].r>>1;
    if (x<=mid) change(rt<<1,x,v);
    else change(rt<<1|1,x,v);
    update(rt);
}
int main()
{
    n=read();d=read();
    for (int i=1;i<=n;i++) a[i]=read();
    build(1,1,n);
    for (int i=1;i<=d;i++)
    {
        int x=read(),v=read();
        change(1,x,v);
        ll p=0;
        for (int j=0;j<=1;j++)
            for (int k=0;k<=1;k++)
                p=max(p,t[1].f[j][k]);
        ans+=1ll*p;
    }
    printf("%lld\n",ans);
    return 0;
}

 

posted @ 2018-07-22 20:37  星星之火OIer  阅读(252)  评论(0编辑  收藏  举报