BZOJ1915[USACO 2010 Open Gold 1.Cow Hopscotch]——DP+斜率优化

BZOJ1915[USACO 2010 Open Gold 1.Cow Hopscotch]——DP+斜率优化

大家好,又见面了,我是你们的朋友全栈君。

题目描述

奶牛们正在回味童年,玩一个类似跳格子的游戏,在这个游戏里,奶牛们在草地上画了一行N个格子,(3 <=N <= 250,000),编号为1..N。就像任何一个好游戏一样,这样的跳格子游戏也有奖励!第i个格子标有一个数字V_i(-2,000,000,000 <=V_i <= 2,000,000,000)表示这个格子的钱。奶牛们想看看最后谁能得到最多的钱。规则很简单: * 每个奶牛从0号格子出发。(0号格子在1号之前,那里没钱) * 她向N号格子进行一系列的跳跃(也可以不跳),每次她跳到的格子最多可以和前一 个落脚的格子差K格(1 <= K <= N)(比方说,当前在1号格,K=2, 可以跳到2号和3号格子) *在任何时候,她都可以选择回头往0号格子跳,直到跳到0号格子。另外,除了以上规则之外,回头跳的时候还有两条规则: *不可以跳到之前停留的格子。 *除了0号格子之外,她在回来的时候,停留的格子必须是恰巧过去的时候停留的某个格子的前一格(当然,也可以跳过某些过去时候停留的格子)。简单点说,如果i号格子是回来 停留的格子,i+1号就必须是过去停留的格子,如果i+1号格子是过去停留的格子,i号格子不一定要是回来停留的格子。(如果这里不太清楚的可以去看英文原文)她得到的钱就是所有停留过的格子中的数字的和,请你求出最多奶牛可以得到的钱数。 在样例中,K=2,一行5个格子。
BZOJ1915[USACO 2010 Open Gold 1.Cow Hopscotch]——DP+斜率优化 一个合法的序列Bessie可以选择的是0[0], 1[0], 3[2], 2[1], 0[0]。(括号里的数表示钱数) 这样,可以得到的钱数为0+0+2+1+0 = 3。 如果Bessie选择一个序列开头为0, 1, 2, 3, …,那么她就没办法跳回去了,因为她没办法再跳到一个之前没跳过的格子。序列0[0], 2[1], 4[-3], 5[4], 3[2], 1[0], 0[0]是最大化钱数的序列之一,最后的钱数为(0+1-3+4+2+0 = 4)。

输入

* 第1行 1: 两个用空格隔开的整数: N 和 K * 第2到N+1行: 第i+1行有一个整数: V_i

输出

* 第一行: 一个单个的整数表示最大的钱数是多少。

样例输入

5 2
0
1
2
-3
4

样例输出

4
OUTPUT DETAILS:
还有一种可能的最大化钱数的序列是: 0 2 4 5 3 1 0
 
考虑到题目叙述可能不太清楚,在这里大致说一下题目大意:奶牛要向前跳格子并在跳到某个格子后要向回跳最终跳回起点,每个格子有一个价值(有正有负),且向前跳时每次最多向前跳K个。在向回跳时每次同样最多跳k个且每次必须跳到去时跳的某个格子的前一个格子,每次跳的不能是去时的格子,求最大获得价值。
询问最大值,考虑贪心、搜索和dp,显然贪心是不行,数据范围搜索也不可过,所以自然想到dp。因为奶牛一定要回去,所以设的dp方程要保证奶牛能回去。因为去和回来所跳距离限制相同,所以去时从x跳到y,回来时一定能从y-1跳到x-1,再结合跳回来的规则不难想出f[i]表示去时当前跳到i且留下i-1作为回来的路所得到的最大价值。因为价值要尽可能大,所以一来一回自然要把i之前的所有正数都跳到,再处理出s[i]表示前i个数中所有正数的和,val[i]表示i点的价值。于是就得出了dp转移方程:
f[i]=max{f[j]+s[i-2]-s[j]}+val[i-1]+val[i],(i-K<=j<i-1)。因为回来时一定要走i-1,所以先把它加上。但答案可不是max{f[i]},因为对于f[i],我们留下了i-1作为回去时的落脚点,所以我们还可以把[i+1,i-1+K]中所有正数点走完,最后的结果就是max{f[i]+s[i-1+K]-s[i]}。因为f[i]的转移只和i,j有关所以可以斜率优化。
最后附上代码。
#include<queue>
#include<cmath>
#include<cstdio>
#include<cstring>
#include<iostream>
#include<algorithm>
using namespace std;
long long f[250010];
long long s[250010];
int v[250010];
int n,m;
int q[250010];
int l,r;
int main()
{
    scanf("%d%d",&n,&m);
    for(int i=1;i<=n;i++)
    {
        scanf("%d",&v[i]);
        if(v[i]>0)
        {
            s[i]=s[i-1]+v[i];
        }
        else
        {
            s[i]=s[i-1];
        }
        f[i]=-1ll<<60;
    }
    f[0]=0;
    l=r=1;
    for(int i=2;i<=n;i++)
    {
        while(l<=r&&q[l]<i-m)
        {
            l++;
        }
        f[i]=f[q[l]]+s[i-2]+v[i-1]+v[i]-s[q[l]];
        while(l<=r&&f[q[r]]-s[q[r]]<f[i-1]-s[i-1])
        {
            r--;
        }
        q[++r]=i-1;
    }
    long long ans=s[m];
    for(int i=1;i<=n;i++)
    {
        if(i+m-1<=n)
        {
            ans=max(ans,f[i]+s[i+m-1]-s[i]);
        }
        else
        {
            ans=max(ans,f[i]+s[n]-s[i]);
        }
    }
    printf("%lld",ans);
}

转载于:https://www.cnblogs.com/Khada-Jhin/p/9157529.html

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/107481.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)
blank

相关推荐

  • PHP工厂模式学习「建议收藏」

    PHP工厂模式学习「建议收藏」PHP工厂模式学习目录PHP工厂模式学习介绍引入定义参与者工厂方法模式实例工厂方法模式的优点工厂方法模式缺点适用环境小结介绍工厂方法模式,可以更好的处理客户的需求变化。引入简单工厂模式把实例化对象的工作推迟到了专门的工厂类中。但是当客户需求出现变化的时候,我们不仅得增加新的类和修改工厂类以适应客户的需求,这是设计模式所不允许的。工厂方法模式应运而生。解决思路:那里变化,封装哪里

  • 国内CMS技术发展的外在表现形式「建议收藏」

    国内CMS技术发展的外在表现形式「建议收藏」网站作为企事业单位的网上名片已经成为必需。大多数单位都选择利用成熟的CMS(内容管理系统)交给专业的公司或者本单位负责人(相当于站长)建设自己的网站。国内CMS技术的不断发展的外在表现,以便站长和建站公司选择合适的CMS系统建设更强大的网站。我建站时用过很多CMS系统,各有各的特点。现在之所以选择主要用We7的CMS是觉得在以下方面还是不错的。一产品成熟度。据我了解We7CMS系统

  • java标识符可以$开头吗_JAVA标识符

    java标识符可以$开头吗_JAVA标识符JAVA标识符JAVA标识符简介Java语言中,对于变量,常量,函数,语句块也有名字,我们统统称之为Java标识符。也就是程序员在定义java程序时,自定义的一些名字,例如helloworld程序里关键字class后跟的Demo,就是我们定义的类名。类名就属于标识符的一种。标识符除了应用在类名上,还可以用在变量、函数名、包名上。(要求同学们先记住,以后会详细见到这些)。标识符命名规则1.标识符由…

  • Vue学习之从入门到神经(两万字收藏篇)

    Vue学习之从入门到神经(两万字收藏篇)Vue史诗级教程系列文章,欢迎订阅专栏

  • android singleTask

    android singleTask本文载自http://blog.csdn.net/wang_zun_ren/article/details/6823257现有2个项目,taskA、taskB。taskA负责调用taskB中指定的界面。taskB中有3个界面,a、b、c,每个界面显示它所在的taskid。SingleTask:其中b界面被声明为SingleTask。先运行tas

  • spring boot 系列之三:spring boot 整合JdbcTemplate

    前面两篇文章我们讲了两件事情:这篇文章我们来看下怎么通过JdbcTemplate进行数据的持久化。废话不多说,直接上干货。一、代码实现packagecom.study.entity;pub

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号