洛谷P1198 [JSOI2008]最大数(BZOJ.1012 )
题目描述
现在请求你维护一个数列,要求提供以下两种操作:
1、 查询操作。
语法:Q L
功能:查询当前数列中末尾L个数中的最大的数,并输出这个数的值。
限制:L不超过当前数列的长度。
2、 插入操作。
语法:A n
功能:将n加上t,其中t是最近一次查询操作的答案(如果还未执行过查询操作,则t=0),并将所得结果对一个固定的常数D取模,将所得答案插入到数列的末尾。
限制:n是整数(可能为负数)并且在长整范围内。
注意:初始时数列是空的,没有一个数。
输入输出格式
输入格式:
第一行两个整数,M和D,其中M表示操作的个数(M <= 200,000),D如上文中所述,满足(0<D<2,000,000,000)
接下来的M行,每行一个字符串,描述一个具体的操作。语法如上文所述。
输出格式:
对于每一个查询操作,你应该按照顺序依次输出结果,每个结果占一行。
输入输出样例
5 100
A 96
Q 1
A 97
Q 1
Q 2
96
93
96
说明
[JSOI2008]
思路:
1.线段树。在初始建树时将所有节点赋值为-INF,操作和单点修改与区间最值并没有多少不同。
2.单调队列+二分查找。
代码:
1.线段树 722ms/17.06MB,比方法2要慢近一倍。
#include<cstdio>
#include<algorithm>
#define LL long long
using namespace std;
const int N=,INF=1e9+; int m,mod,len;
LL t,Max[N<<]; void read(int &now)
{
now=;bool f=;char c=getchar();
while(c<''||c>'')
{
if(c=='-')f=;
c=getchar();
}
while(c>=''&&c<='')now=(now<<)+(now<<)+c-'',c=getchar();
now= f?-now:now;
} void PushUp(int rt)
{
Max[rt]= Max[rt<<]<Max[rt<<|]?Max[rt<<|]:Max[rt<<];
} void Build(int l,int r,int rt)
{
if(l==r)
{
Max[rt]=-INF;
return;
}
int m=(l+r)>>;
Build(l,m,rt<<);
Build(m+,r,rt<<|);
PushUp(rt);
} void ModifyPoint(int l,int r,int rt,int p,int v)
{
if(l==r)
{
Max[rt]= Max[rt]<v?v:Max[rt];
return;
}
int m=(l+r)>>;
if(p<=m) ModifyPoint(l,m,rt<<,p,v);
else ModifyPoint(m+,r,rt<<|,p,v);
PushUp(rt);
} LL QueryMax(int l,int r,int rt,int L,int R)
{
if(L<=l && r<=R) return Max[rt];
int m=(l+r)>>;
LL res=-INF;
if(L<=m) res=max(res,QueryMax(l,m,rt<<,L,R));
if(m<R) res=max(res,QueryMax(m+,r,rt<<|,L,R));
return res;
} int main()
{
read(m);read(mod);
Build(,m,);
for(int i=;i<=m;++i)
{
char opt[];int num;
scanf("%s",opt);read(num);
if(opt[]=='A')
ModifyPoint(,m,,++len,(num+t)%mod);
else
printf("%lld\n",t=QueryMax(,m,,len-num+,len));
}
return ;
}
线段树(插入元素 区间最值)
2.单调队列+lower_bound(+栈) 337ms/12.6MB
(先写 if(opt[0]=='Q')+询问 后写插入 就会错是什么鬼。。 哪位dalao看出来帮我指出一下。。thanks)
#include<cstdio>
#include<algorithm>
using namespace std;
const int N=; int m,mod,len,size,t,Num[N],Stack[N]; void read(int &now)
{
now=;bool f=;char c=getchar();
while(c<''||c>'')
{
if(c=='-')f=;
c=getchar();
}
while(c>=''&&c<='')now=(now<<)+(now<<)+c-'',c=getchar();
now= f?-now:now;
} /*int Mylower_bound(int s,int t,int p)
{
int l=s,r=t,m;
while(l<r)
{
m=(l+r)>>1;
if(Stack[m]<p)
l=m+1;
else if(Stack[m]==p)
return m;
else
r=m;
}
return l;
}*/ int main()
{
read(m);read(mod);
char opt[];int a;
while(m--)
{
scanf("%s",opt);read(a);
if(opt[]=='A')
{//因为比当前元素小的数在最后的求最大值中毫无作用,所以直接弹出
a=(a+t)%mod;
Num[++len]=a;
while(size && Num[Stack[size]]<=a)
--size;
Stack[++size]=len;
}
else
{//Stack[i]存储的是一个位置pos,且这些位置单调递增,这些位置对应的Num[pos]也单调递增
int pos=lower_bound(Stack+,Stack+size+,len-a+)-Stack;//找出一个Stack中大于等于查询位置的pos
//int pos=Mylower_bound(1,size,len-a+1);
printf("%d\n",t=Num[Stack[pos]]);
}
}
return ;
}
单调队列
错误的写法
#include<cstdio>
#include<algorithm>
using namespace std;
const int N=; int m,mod,len,size,t,Num[N],Stack[N]; void read(int &now)
{
now=;bool f=;char c=getchar();
while(c<''||c>'')
{
if(c=='-')f=;
c=getchar();
}
while(c>=''&&c<='')now=(now<<)+(now<<)+c-'',c=getchar();
now= f?-now:now;
} int main()
{
read(m);read(mod);
char opt[];int a;
while(m--)
{
scanf("%s",opt);read(a);
if(opt[]=='Q')
{//Stack[i]存储的是一个位置,且这些位置单调递增,这些位置对应的Num[pos]也单调递增
int pos=lower_bound(Stack+,Stack+size+,len-a+)-Stack;
printf("%d\n",t=Num[pos]);
}
else
{//因为比当前元素小的数在最后的求最大值中毫无作用,所以直接弹出
a=(a+t)%mod;
Num[++len]=a;
while(size && Num[Stack[size]]<=a)
--size;
Stack[++size]=len;
}
}
return ;
}
0分
洛谷P1198 [JSOI2008]最大数(BZOJ.1012 )的更多相关文章
- 洛谷 P1198 [JSOI2008]最大数
洛谷 P1198 [JSOI2008]最大数 题目描述 现在请求你维护一个数列,要求提供以下两种操作: 1. 查询操作. 语法:Q L 功能:查询当前数列中末尾L个数中的最大的数,并输出这个数的值. ...
- 洛谷P1198 [JSOI2008]最大数(单点修改,区间查询)
洛谷P1198 [JSOI2008]最大数 简单的线段树单点问题. 问题:读入A和Q时,按照读入一个字符会MLE,换成读入字符串就可以了. #include<bits/stdc++.h> ...
- 「线段树」「单点修改」洛谷P1198 [JSOI2008]最大数
「线段树」「单点修改」洛谷P1198 [JSOI2008]最大数 题面描述 现在请求你维护一个数列,要求提供以下两种操作: 1. 查询操作. 语法:Q L 功能:查询当前数列中末尾L个数中的最大的数, ...
- 洛谷P1198 [JSOI2008]最大数
P1198 [JSOI2008]最大数 267通过 1.2K提交 题目提供者该用户不存在 标签线段树各省省选 难度提高+/省选- 提交该题 讨论 题解 记录 最新讨论 WA80的戳这QwQ BZOJ都 ...
- 【题解】洛谷P1198 [JSOI2008] 最大数(线段树)
洛谷P1198:https://www.luogu.org/problemnew/show/P1198 思路 一道水水的线段树 20分钟A掉 这道题只涉及到单点修改和区间查询 所以这道题甚至不用Laz ...
- BZOJ——1012: [JSOI2008]最大数maxnumber || 洛谷—— P1198 [JSOI2008]最大数
http://www.lydsy.com/JudgeOnline/problem.php?id=1012|| https://www.luogu.org/problem/show?pid=1198 T ...
- 洛谷 P1198 [JSOI2008]最大数 Label:线段树
题目描述 现在请求你维护一个数列,要求提供以下两种操作: 1. 查询操作. 语法:Q L 功能:查询当前数列中末尾L个数中的最大的数,并输出这个数的值. 限制:L不超过当前数列的长度. 2. 插入操作 ...
- 洛谷 P1198 [JSOI2008]最大数——单调栈/线段树
先上一波题目 https://www.luogu.org/problem/P1198 题目要求维护后缀最大值 以及在数列的最后面添加一个数 这道题呢我们有两种做法 1.单调栈 因为只需要维护后缀最大值 ...
- 洛谷P1198 [JSOI2008]最大数(线段树/单调栈)
题目链接: https://www.luogu.org/problemnew/show/P1198 题目描述 现在请求你维护一个数列,要求提供以下两种操作: 1. 查询操作. 语法:Q L 功能:查询 ...
随机推荐
- 【转】Shell编程基础篇-下
[转]Shell编程基础篇-下 1.1 条件表达式 1.1.1 文件判断 常用文件测试操作符 常用文件测试操作符 说明 -d文件,d的全拼为directory 文件存在且为目录则为真,即测试表达式成立 ...
- Django 查询集简述
通过模型中的管理器构造一个查询集(QuerySet),来从数据库中获取对象.查询集表示从数据库中取出来的对象的集合.它可以含有零个.一个或者多个过滤器.过滤器基于所给的参数限制查询的结果. 从SQL ...
- Linux内核调试 - 一般人儿我都不告诉他(一)【转】
转自:http://www.cnblogs.com/armlinux/archive/2011/04/14/2396821.html 悄悄地进入Linux内核调试(一) 本文基址:http://blo ...
- dubbo系列二、dubbo+zookeeper+dubboadmin分布式服务框架搭建(windows平台)
一.zookeeper配置中心安装 1.下载安装包,zookeeper-3.4.6.tar.gz 2.解压安装包,修改配置文件 参考zookeeper-3.4.6/conf/zoo_sample.cf ...
- python通过操作windows系统注册表方式修改环境变量
#coding=utf8 import os import sys from subprocess import check_call if sys.hexversion > 0x0300000 ...
- CSS导航条nav简单样式
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
- Android软件更新
Android软件更新 //得到当前版本编码和版本名称. public static int getVerCode(Context context) { ; try { verCode =).vers ...
- jquery load加载页面内ajax返回的div不能响应页面js的问题的解决方案
1. 前言 由于项目需要,需要load一个页面并保持ajax返回的div能响应其页面内的JS的click事件.这个不是 解决用jquery load加载页面到div时,不执行页面js的问题 这类问题, ...
- Java 开发环境配置--eclipse工具进行java开发
Java 开发环境配置 在本章节中我们将为大家介绍如何搭建Java开发环境. Windows 上安装开发环境 Linux 上安装开发环境 安装 Eclipse 运行 Java Cloud Studio ...
- lodash篇之对象深度比较_.isEqual