Sequence(ST表)(洛谷P2048)
知识储备
在做这道题前,我们先要了解一下ST表(一种离线求区间最值的方法)
ST表使用DP实现的,其查询复杂度为O(1).
那么我们怎么用DP实现呢??
首先,我们设立一个状态f[i][j],其中i代表起点(包括在内),而j则代表是2的几次方,那么这个状态的含义就是以i为起点,求i~i+2^j-1内的区间最大值
最开始,我们定义f[i][0]的值为其本身即a[i]
那么我们转移方程就是
f[i][j]=max(f[i][j-1],f[i+(1<<(j-1)][j-1]); //一层一层的推过来...
然后我们就知道在一定区间内的最大值了!!
查询也是一个需要特别注意的点,我们要知道查询的x位置和y位置之间max,就要知道y-x+1是y的几次方,其中这个要向下取整,防止越界,但是这样我们就会有一部分扫不到,由于我们向下取整的这个区间是大于等于这个区间的一半的,所以我们可以反过来,从y-(1<<k)到y,这样可以保证不会漏掉数据,并且重复的话也不会对答案产生影响
但是注意!!算k的式子要小心啊
别取错了,一个小BUG就会导致你WA掉...
#include<bits/stdc++.h>
#define ll long long
#define FOR(i,a,b) for(register int i=a;i<=b;i++)
#define ROF(i,a,b) for(register int i=a;i>=b;i--)
using namespace std;
const int N=+;
int n,m;
int a[N];
int f[N][];
int scan()
{
int as=,f=;
char c=getchar();
while(c>''||c<''){if(c=='-') f=-;c=getchar();}
while(c>=''&&c<=''){as=(as<<)+(as<<)+c-'';c=getchar();}
return as*f;
}
void RMQ(int n)
{
FOR(i,,)
{
FOR(j,,n)
{
if(j+(<<i)-<=n)
f[j][i]=max(f[j][i-],f[j+(<<(i-))][i-]);
}
}
}
int main()
{
n=scan();
m=scan();
FOR(i,,n)
a[i]=scan(),f[i][]=a[i];
RMQ(n);
FOR(i,,m)
{
int x,y;
x=scan();
y=scan();
int k=(int)(log((double)(y-x+))/log(2.0));
printf("%d\n",max(f[x][k],f[y-(<<k)+][k]));
}
return ;
}
代码戳这里
接下来我们要讲正事了!!
我们的目光转到钢琴这一题上
思路
首先,我们前缀和储存值,然后在用KMQ,将一些区间内的max求出来
然后我们根据题目要求,由于题目要我们求最大权值,那么我们就求每个可能区间里面的max,将其储存在堆里面.
但是最开始的时候,我们只求出了以某一点为起点的最大值,但是我们最后要求的1~k个最大值的相加
所以我们在每取出一个max是,再去寻找以该点为起点的max,只不过此时的max不是当前的最有解,而是次优解,但是也可能成为新的最有解,
然后题目打这里就差不多啦
#include<bits/stdc++.h>
#define ll long long
#define gc getchar()
#define FOR(i,a,b) for(register int i=a;i<=b;i++)
#define ROF(i,a,b) for(register int i=a;i>=b;i--)
using namespace std;
const int N=+;
int n,m,L,R;
ll sum[N];
int id[N][],f[N][];
ll ans=;
int scan()
{
int as=,f=;char c=gc;
while(c>''||c<''){if(c=='-') f=-;c=gc;}
while(c>=''&&c<=''){as=(as<<)+(as<<)+c-'';c=gc;}
return as*f;
}
//learn new
struct s1
{
int u,l,r,pos;
bool operator <(s1 y) const
{
return sum[pos]-sum[u-]<sum[y.pos]-sum[y.u-];//大的放队尾
}
}tmp;
priority_queue<s1>q;//队列
void init(int &pos,int l,int r)
{
int j=log2(r-l+);
if(f[l][j]>f[r-(<<j)+][j]) pos=id[l][j];
else pos=id[r-(<<j)+][j];
}//这里都懂吧,KMQ的查询
int main()
{
n=scan();m=scan();L=scan();R=scan();
FOR(i,,n)
{
sum[i]=scan();sum[i]+=sum[i-];//前缀和
f[i][]=sum[i];
id[i][]=i;//位置的处理
}
FOR(j,,)
{
FOR(i,,n)
{
if(i+(<<j)->n) break;
if(f[i][j-]>f[i+(<<(j-))][j-]) f[i][j]=f[i][j-],id[i][j]=id[i][j-];
else f[i][j]=f[i+(<<(j-))][j-],id[i][j]=id[i+(<<(j-))][j-];
}//这里其实就是KMQQAQ
}
FOR(i,,n-L+)//入坑silasila......
{
int l=i+L-; int r=min(i+R-,n),pos=,u=i;//随便给pos一个值
init(pos,l,r);//pos在这里赋好了值
tmp=(s1){u,l,r,pos};//然后加入四元组吧..
q.push(tmp);//入队
}
while(m--)
{
tmp=q.top(); q.pop();//弹出顶端,即最优解
int u=tmp.u,l=tmp.l,r=tmp.r,tps=tmp.pos,pos=tmp.pos;//去寻找次
//优解
ans+=sum[tps]-sum[u-];
if(tps>l)
{
init(pos,l,tps-);tmp=(s1){u,l,tps-,pos};q.push(tmp);
}
if(tps<r)
{
init(pos,tps+,r);tmp=(s1){u,tps+,r,pos};q.push(tmp);
}
}
printf("%lld",ans);
return ;
}
代码戳这里
Sequence(ST表)(洛谷P2048)的更多相关文章
- 【模板】ST表 洛谷P1816 忠诚
P1816 忠诚 题目描述 老管家是一个聪明能干的人.他为财主工作了整整10年,财主为了让自已账目更加清楚.要求管家每天记k次账,由于 管家聪明能干,因而管家总是让财主十分满意.但是由于一些人的挑拨, ...
- 洛谷 P2048 [NOI2010]超级钢琴 解题报告
P2048 [NOI2010]超级钢琴 题目描述 小Z是一个小有名气的钢琴家,最近C博士送给了小Z一架超级钢琴,小Z希望能够用这架钢琴创作出世界上最美妙的音乐. 这架超级钢琴可以弹奏出n个音符,编号为 ...
- [洛谷P2048] [NOI2010] 超级钢琴
洛谷题目链接:[NOI2010]超级钢琴 题目描述 小Z是一个小有名气的钢琴家,最近C博士送给了小Z一架超级钢琴,小Z希望能够用这架钢琴创作出世界上最美妙的音乐. 这架超级钢琴可以弹奏出n个音符,编号 ...
- 洛谷 P2048 [NOI2010]超级钢琴 || Fantasy
https://www.luogu.org/problemnew/show/P2048 http://www.lydsy.com/JudgeOnline/problem.php?id=2006 首先计 ...
- 洛谷 P2048 BZOJ 2006 [NOI2010]超级钢琴
题目描述 小Z是一个小有名气的钢琴家,最近C博士送给了小Z一架超级钢琴,小Z希望能够用这架钢琴创作出世界上最美妙的音乐. 这架超级钢琴可以弹奏出n个音符,编号为1至n.第i个音符的美妙度为Ai,其中A ...
- 洛谷 P2048 [NOI2010]超级钢琴(优先队列,RMQ)
传送门 我们定义$(p,l,r)=max\{sum[t]-sum[p-1],p+l-1\leq t\leq p+r-1 \}$ 那么因为对每一个$p$来说$sum[p-1]$是一个定值,所以我们只要在 ...
- 洛谷P2048 [NOI2010]超级钢琴 题解
2019/11/14 更新日志: 近期发现这篇题解有点烂,更新一下,删繁就简,详细重点.代码多加了注释.就酱紫啦! 正解步骤 我们需要先算美妙度的前缀和,并初始化RMQ. 循环 \(i\) 从 \(1 ...
- 【洛谷3865】 【模板】ST表(猫树)
传送门 洛谷 Solution 实测跑的比ST表快!!! 这个东西也是\(O(1)\)的,不会可以看我上一篇Blog 代码实现 代码戳这里
- 【洛谷】P2880 [USACO07JAN]平衡的阵容Balanced Lineup(st表)
题目背景 题目描述: 每天,农夫 John 的N(1 <= N <= 50,000)头牛总是按同一序列排队. 有一天, John 决定让一些牛们玩一场飞盘比赛. 他准备找一群在对列中为置连 ...
随机推荐
- Mac下用tomcat搭建下载服务器
1.下载tomcat 去官方网址: http://tomcat.apache.org/ 下载最新版 2.下载解压后,自己可以随便放在哪个文件夹下,自己记得路径即可.比如Users/你的用户名/Docu ...
- Python中该使用%还是format来格式化字符串?
%还是format 1.皇城PK Python中格式化字符串目前有两种阵营:%和format,我们应该选择哪种呢? 自从Python2.6引入了format这个格式化字符串的方法之后,我认为%还是fo ...
- VIN码识别:助力汽车后市场转型升级
随着中国汽车市场的成熟,汽车后市场发展迅速,呈“井喷”式增长.据最新数据统计,2015年,中国汽车后市场产值突破8000亿规模,到2018年有望突破万亿. 所谓汽车后市场是指汽车销售以后,围绕汽车使用 ...
- flask中static_folder与static_url_path的区别与联系
# -*- coding:utf-8 -*- from flask import Flask, url_for app1 = Flask(__name__, static_folder='mystat ...
- The Erdös-Straus Conjecture 题解
题面 Description The Brocard Erdös-Straus conjecture is that for any integern > 2 , there are posit ...
- CodeForces-455A Boredom
题目链接 https://vjudge.net/problem/CodeForces-455A 题面 Description Alex doesn't like boredom. That's why ...
- 给移动硬盘安装rhel7
本机是win8.1的系统,但不想给电脑装双系统,所以想给移动硬盘里安装rhel7移动硬盘是750G的在网上搜了很多方法,我采取了两个方法:方法一.1.取一个U盘,用软碟通把rhel7的iso文件写进了 ...
- 使用ZSetOperations(有序)操作redis
方法 c参数 s说明 Boolean add(K key, V value, double score); K key:集合key V value:key对应的值 double score:分数 向 ...
- CentOS6.8单独编译安装PHP gd库扩展
# PHP-GD安装 #在安装之前可以先更新一下yum源,可以使用国内的阿里云源 yum -y install libjpeg-turbo-devel yum -y install freetype- ...
- [OS] 进程相关知识点
进程概念: 1.程序在执行中 2.一个具有一定独立功能的程序在一个数据集合上的一次动态执行过程,是系统进行资源分配和调度的独立单位. 进程与程序的差别: ·进程----动态, 程序----静态 ·进程 ...