超级钢琴

知识储备

在做这道题前,我们先要了解一下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)的更多相关文章

  1. 【模板】ST表 洛谷P1816 忠诚

    P1816 忠诚 题目描述 老管家是一个聪明能干的人.他为财主工作了整整10年,财主为了让自已账目更加清楚.要求管家每天记k次账,由于 管家聪明能干,因而管家总是让财主十分满意.但是由于一些人的挑拨, ...

  2. 洛谷 P2048 [NOI2010]超级钢琴 解题报告

    P2048 [NOI2010]超级钢琴 题目描述 小Z是一个小有名气的钢琴家,最近C博士送给了小Z一架超级钢琴,小Z希望能够用这架钢琴创作出世界上最美妙的音乐. 这架超级钢琴可以弹奏出n个音符,编号为 ...

  3. [洛谷P2048] [NOI2010] 超级钢琴

    洛谷题目链接:[NOI2010]超级钢琴 题目描述 小Z是一个小有名气的钢琴家,最近C博士送给了小Z一架超级钢琴,小Z希望能够用这架钢琴创作出世界上最美妙的音乐. 这架超级钢琴可以弹奏出n个音符,编号 ...

  4. 洛谷 P2048 [NOI2010]超级钢琴 || Fantasy

    https://www.luogu.org/problemnew/show/P2048 http://www.lydsy.com/JudgeOnline/problem.php?id=2006 首先计 ...

  5. 洛谷 P2048 BZOJ 2006 [NOI2010]超级钢琴

    题目描述 小Z是一个小有名气的钢琴家,最近C博士送给了小Z一架超级钢琴,小Z希望能够用这架钢琴创作出世界上最美妙的音乐. 这架超级钢琴可以弹奏出n个音符,编号为1至n.第i个音符的美妙度为Ai,其中A ...

  6. 洛谷 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]$是一个定值,所以我们只要在 ...

  7. 洛谷P2048 [NOI2010]超级钢琴 题解

    2019/11/14 更新日志: 近期发现这篇题解有点烂,更新一下,删繁就简,详细重点.代码多加了注释.就酱紫啦! 正解步骤 我们需要先算美妙度的前缀和,并初始化RMQ. 循环 \(i\) 从 \(1 ...

  8. 【洛谷3865】 【模板】ST表(猫树)

    传送门 洛谷 Solution 实测跑的比ST表快!!! 这个东西也是\(O(1)\)的,不会可以看我上一篇Blog 代码实现 代码戳这里

  9. 【洛谷】P2880 [USACO07JAN]平衡的阵容Balanced Lineup(st表)

    题目背景 题目描述: 每天,农夫 John 的N(1 <= N <= 50,000)头牛总是按同一序列排队. 有一天, John 决定让一些牛们玩一场飞盘比赛. 他准备找一群在对列中为置连 ...

随机推荐

  1. 《剑指offer》题解

    有段时间准备找工作,囫囵吞枣地做了<剑指offer>提供的编程习题,下面是题解收集. 当初没写目录真是个坏习惯(-_-)||,自己写的东西都要到处找. 提交的源码可以在此repo中找到:h ...

  2. 项目总结(一)->项目的七宗罪

    大半夜来这一份总结,心中夹杂着各种各样的心情,酸甜苦辣都有,今天为止,整个项目终于完结了,对于这样一个本可以正而八经吃吃薯片,看看毛片就可以完成项目,最后演变成一个一月之内连续加班105个小时的项目, ...

  3. 把python脚本打包成win可执行文件

    前几天有个朋友找我写一点小东西,写好后把代码发他帮他搞了半天,结果愣是没听懂,就找到了这个办法. 1.导入pyinstaller包, pip install pyinstaller 2.进入到你需要打 ...

  4. python中socket、socketio、flask-socketio、WebSocket的区别与联系

    socket.socketio.flask-socketio.WebSocket的区别与联系 socket 是通信的基础,并不是一个协议,Socket是应用层与TCP/IP协议族通信的中间软件抽象层, ...

  5. 51单片机实现定时器中断0-F

    #include <reg51.h> #define uint unsigned int #define uchar unsigned char sfr P0M0 = 0x94; sfr ...

  6. Halcon17对硬件配置要求

     Halcon17对硬件配置要求 Halcon17已经发布出来了,很多朋友一定想安装这款机器视觉软件来学习,我们今天给大家讲解下,Halcon17对硬件配置的要求: Halcon17 For Wind ...

  7. 小程序开发时PC端调试返回结果和手机端IOS不一致问题

    IOS11登录时遇到一个请求与PC返回不一致情况, 在小程序调试时IOS上始终没有wx.request() 不能发送请求 尝试解决方法 打开微信小程序调试的设置, 将TLS设为可信任的域名 设置 -- ...

  8. rcnn spp_net

    在http://www.cnblogs.com/jianyingzhou/p/4086578.html中 提到 rcnn开创性工作,但是计算时间太长,重复计算太大. spp_net将重复计算避免了 我 ...

  9. Reactor模式是什么(转载)

    一.Reactor模式是什么反应器设计模式(Reactor pattern)是一种为处理并发服务请求,并将请求提交到一个或者多个服务处理程序的事件设计模式.当客户端请求抵达后,服务处理程序使用多路分配 ...

  10. DataGridView加载gif图片

    当我们想加载图片时,一般情况下都会使用picturebox控件,这个控件可以加载各种格式的图片,当然也包括gif图片.但是有时,我们也希望一些数据展示控件也可以加载图片,比如说DataGridView ...