★实验任务

小 F 很爱打怪,今天因为系统 bug,他提前得知了 n 只怪的出现顺序以及击 倒每只怪得到的成就值 ai。设第一只怪出现的时间为第 1 秒,这个游戏每过 1 秒 钟出现一只新怪且没被击倒的旧怪消失。小 F 决定发动一次技能,他的技能最多 维持 k 秒,他希望获得最大的成就值,请你帮他计算他发动技能的时间 l 和技能 结束时间 r(r-l+1<=k)。当存在多种方案使得成就值最大时,选择技能发动时间 l 最小的方案,再选择技能持续时间 r-l+1 最小的方案。

★数据输入

输入第一行为两个正整数 n(1<=n<=100000),k(0<k<=n),表示出现 n 只怪, 小 F 的技能最多维持 k 秒。 输入第二行为 n 个整数,表示小 F 击倒第 i 秒钟出现的怪能给有获得的成就 值 ai(-1000<=a[i]<=1000)。

★数据输出

输出为一行三个数。第一个数为可获得的最大成就值,第二个数为技能发动 时间 l,第三个数为技能结束时间 r。

输入示例 输出示例
6 3
-1 2 -6 5 -5 6
6 4 6
输入示例 输出示例
5 5
-1 -1 -1 -1 -1
-1 1 1

★思路

这道题的目标就是找到长度不超过k的最大子序列和,求子序列和可以使用前缀和与后缀和的思想。如果先使用前缀和做法,把数组倒序就是后缀和做法

前缀和的概念就是第i个数前面所有数字的和。所以[l,r]区间的和可以写成。(sum[r]-sum[l])。



这样子要找到数字i前面不超过k长度的最大子序列和,就要找到它前面k个前缀和的最小值。(因为sum[i]为定值)。

要找到sum[i]前面k个前缀和的最小值做法是,用递增队列。

建立结构体数组来模拟这个过程。

struct intt
{
int value;
int pos; //记录位置,判断长度是否超过k
int start;
};

首先队首元素存0(因为sum[1]前面没以有元素,所以sum[0]=0)。

接下来对于每一个sum[i],先用它去减队首元素(队首元素代表第i个数前面的最小前缀和)。然后把它加入队列。

如果它比它前面的数字值小,就把它前面的元素弹出(这相当于更新当前的最小值)。

如果它比它前面的数字大,就把它留起来备用。因为他前面的数字最多可以使用k次,如果他前面元素使用k次被弹出后,就要用它来补充当前最小前缀和这个位置了。

如果它与前面元素相等的话,也要弹出前面元素,因为前面元素已经被使用它过一次了,而它没有被使用过。

这样子我们找到的是每个数字i所对应的起始位置比较靠后的满足条件的最大子区间和。如果我们把数组倒序进行这个操作那么我们就找到了满足题目要求的“当存在多种方案使得成就值最大时,选择技能发动时间 l 最小的方案,再选择技能持续时间 r-l+1 最小的方案”。

★Code

 

#include<iostream>
#include<string.h>
using namespace std;
struct intt
{
int value;
int pos;
int start;
};
int a[100005]; //普通的数组
intt dp[100005]; //模拟队列,队首元素代表最小前缀和
intt sum[100005]; //前缀和数组
intt mmax[100005]; //以i结尾符合条件的区间最大和
int main()
{
int n, k, i;
intt *head = &dp[0];
intt *tail = &dp[0];
scanf("%d %d",&n,&k);//cin >> n >> k;
memset(a, 0, sizeof(a));
memset(dp, 0, sizeof(dp));
memset(sum, 0, sizeof(sum));
memset(mmax, 0, sizeof(mmax)); for (i = n; i >= 1; i--)
{
scanf("%d",&a[i]);// cin >> a[i];
}
for (i = 1; i <= n; i++)
{
sum[i].value = sum[i - 1].value + a[i];
sum[i].pos = i;
}
for (i = 1; i <= n; i++)
{
if (tail->pos - head->pos >= k)
head++;
mmax[i].value = sum[i].value - head->value;
mmax[i].start = head->pos;
// cout<<mmax[i].start<<endl;
while (1)
{
if (sum[i].value>tail->value)
{
tail++;
tail->value = sum[i].value;
tail->pos = i;
break;
}
else if (sum[i].value == tail->value)
{
tail->pos = i;
break;
}
else
{
if (tail == head)
{
tail->value = sum[i].value;
tail->pos = sum[i].pos;
break;
}
else
{
tail->value = 0;
tail->pos = 0;
tail--;
}
}
}
}
/*for(int i=1;i<=n;i++)
{
cout<<a[i]<<" ";
}
cout<<endl;
for(int i=1;i<=n;i++)
{
cout<<mmax[i].value<<" ";
}
cout<<endl;*/
int maxx = mmax[1].value;
for (i = 1; i <= n; i++)
{
if (mmax[i].value>maxx)
maxx = mmax[i].value;
}
int temp;
for (i = n; i >= 1; i--)
{
if (mmax[i].value == maxx)
{
temp = i;
break;
}
}
int flag = temp;
int tempp = maxx;
while (1)
{
tempp -= a[flag];
if (!tempp)
break;
flag--;
}
printf("%d %d %d", maxx, n + 1 - temp, n + 1 - flag);
//cout << maxx << " " << n + 1 - temp << " " << n + 1 - flag;
return 0;
}

算法与数据结构实验题 4.2 小 F 打怪的更多相关文章

  1. 算法与数据结构实验题 6.4 Summary

    ★实验任务 可怜的 Bibi 丢了好几台手机以后,看谁都像是小偷,他已经在小本本上记 下了他认为的各个地点的小偷数量. 现在我们将 Bibi 的家附近的地形抽象成一棵有根树.每个地点都是树上的 一个节 ...

  2. 算法与数据结构实验题 4.1 伊姐姐数字 game

    ★实验任务 伊姐姐热衷于各类数字游戏,24 点.2048.数独等轻轻松松毫无压力.一 日,可爱的小姐姐邀请伊姐姐一起玩一种简单的数字 game,游戏规则如下: 一开始桌上放着 n 张数字卡片,从左到右 ...

  3. 算法与数据结构实验题 6.3 search

    ★实验任务 可怜的 Bibi 刚刚回到家,就发现自己的手机丢了,现在他决定回头去搜索 自己的手机. 现在我们假设 Bibi 的家位于一棵二叉树的根部.在 Bibi 的心中,每个节点 都有一个权值 x, ...

  4. 算法与数据结构实验题6.4 order (二叉树)

    1.题目: 2.代码: #include<iostream> #include<algorithm> using namespace std; struct Node { in ...

  5. 算法与数据结构实验题 5.2 Missile

    1.题目: 2.解题思路: 把每个点对应的两条半径求出,之后对d1进行升序排序,对应d2也改变位置.其中一个圆心的半径r1确定之后,除去第一个圆包围的点,在其余点中找到另外一个圆的最长的半径r2,此时 ...

  6. HDU 3791 二叉搜索树 (数据结构与算法实验题 10.2 小明) BST

    传送门:http://acm.hdu.edu.cn/showproblem.php?pid=3791 中文题不说题意. 建立完二叉搜索树后进行前序遍历或者后序遍历判断是否一样就可以了. 跟这次的作业第 ...

  7. 数据结构_sfdg(小F打怪)

    问题描述 小 F 很爱打怪, 今天因为系统 bug, 他提前得知了 n 只怪的出现顺序以及击倒每只怪得到的成就值 ai. 设第一只怪出现的时间为第 1 秒,这个游戏每过 1 秒钟出现一只新怪且没被击倒 ...

  8. 基础算法和数据结构高频题 II

    DFS的两种理解方式:1. 按照实际执行顺序模拟 (适合枚举型DFS,下节课内容)2. 按照DFS的定义宏观理解 (适合分治型DFS,本节课内容) 1 Convert BST to Greater T ...

  9. 第三章 基础算法和数据结构高频题 I

    区间类问题 1 Missing Interval public List<String> findMissingRanges(int[] nums, int lower, int uppe ...

随机推荐

  1. React 父子组件和非父子组件传值

      零.this.props     可以接收到 外界的传值 和 此组件标签内部自定义的方法       例:         <one vals={message} sendVal={this ...

  2. 常用的JavaScript设计模式(二)Factory(工厂)模式

    Factory通过提供一个通用的接口来创建对象,同时,我们还可以指定我们想要创建的对象实例的类型. 假设现在有一个汽车工厂VehicleFactory,支持创建Car和Truck类型的对象实例,现在需 ...

  3. mysql8.0新增用户及密码加密规则修改

    MySQL8.0已经发布GA版,当前最新GA版本为8.0.12.虽然相对于之前版本,MySQL8.0没有加入新元素,但是,经过代码重构,MySQL8.0的优化器更加强大,同时也有一些新特性,如支持索引 ...

  4. 在我的职业生涯中,没有一种技能比 SQL 更有用!

    作者 | Craig Kerstiens 译者 | 阿拉丁 创业公司 CitusData(CitusData 是一家将 PostgreSQL 商业化的初创企业,也是 PostgreSQL 社区领导者, ...

  5. 863. All Nodes Distance K in Binary Tree

    /** * Definition for a binary tree node. * struct TreeNode { * int val; * TreeNode *left; * TreeNode ...

  6. 计算阶乘的和v2.0(4分)

    题目内容: 假设有这样一个三位数m,其百位.十位和个位数字分别是a.b.c,如果m= a!+b!+c!,则这个三位数就称为三位阶乘和数(约定0!=1).请编程计算并输出所有的三位阶乘和数. 函数原型: ...

  7. Vue 生产环境部署

    简要:继上次搭建vue环境后,开始着手vue的学习;为此向大家分享从开发环境部署到生产环境(线上)中遇到的问题和解决办法,希望能够跟各位VUE大神学习探索,如果有不对或者好的建议告知下:*~*! 一. ...

  8. Java设计模式(8)——结构型模式之组合模式(Composite)

    一.概述 定义 将对象以树形结构组织起来,以达成“部分-整体” 的层次结构,使得客户端对单个对象和组合对象的使用具有一致性. 简图 角色——对应上图中顶点为Component,左边为Leaf,右边为C ...

  9. Java基础——注解

    一.概述 引自百度百科: 定义:注解(Annotation),也叫元数据.一种代码级别的说明.它是JDK1.5及以后版本引入的一个特性,与类.接口.枚举是在同一个层次.它可以声明在包.类.字段.方法. ...

  10. [WebService] 使用httpWebrequest 调用并调试WebService

    使用httpWebrequest 调用并调试WebService. 首先  使用httpWebrequest 调用WebService 代码: using System.Net;            ...