Description

IOI国历史研究的第一人——JOI教授,最近获得了一份被认为是古代IOI国的住民写下的日记。JOI教授为了通过这份日记来研究古代IOI国的生活,开始着手调查日记中记载的事件。
日记中记录了连续N天发生的时间,大约每天发生一件。
事件有种类之分。第i天(1<=i<=N)发生的事件的种类用一个整数Xi表示,Xi越大,事件的规模就越大。
JOI教授决定用如下的方法分析这些日记:
1. 选择日记中连续的一些天作为分析的时间段
2. 事件种类t的重要度为t*(这段时间内重要度为t的事件数)
3. 计算出所有事件种类的重要度,输出其中的最大值
现在你被要求制作一个帮助教授分析的程序,每次给出分析的区间,你需要输出重要度的最大值。

Input

第一行两个空格分隔的整数N和Q,表示日记一共记录了N天,询问有Q次。
接下来一行N个空格分隔的整数X1...XN,Xi表示第i天发生的事件的种类
接下来Q行,第i行(1<=i<=Q)有两个空格分隔整数Ai和Bi,表示第i次询问的区间为[Ai,Bi]。

Output

输出Q行,第i行(1<=i<=Q)一个整数,表示第i次询问的最大重要度

Sample Input


Sample Output


HINT

1<=N<=10^5
1<=Q<=10^5
1<=Xi<=10^9 (1<=i<=N)

Source

JOI 2013~2014 春季training合宿 竞技1 By PoPoQQQ

Solution

时限很仁慈,毕竟这题正解就是分块(~常数大的写法慢成狗)

对序列分块,离散化一下,用桶记录出现次数,用块做一个前缀和,再用块的数量的平方的时间跑一个o(x^2)的暴力rmq,就可以预处理所有的初值。

最后对每个询问还是一样的套路,整块的就不枚,直接调用,不完整块的就暴力求解(~你的程序怎么越来越丑(man)了)

//Kaiba_Seto 20170120
//orz cjkmao
#include <math.h>
#include <stdio.h>
#include <memory.h>
#include <algorithm>
#define MaxN 100010
#define MaxS 350
#define RG register
#define inline __inline__ __attribute__((always_inline))
#define L long long
#define dmin(a,b) ((a)<(b)?(a):(b))
#define dmax(a,b) ((a)>(b)?(a):(b)) namespace io{
#define MaxBuf 1<<22
#define _getc() ((S==T&&(T=(S=B)+fread(B,1,MaxBuf,stdin),S==T))?0:*S++)
char B[MaxBuf],*S=B,*T=B;
template<class Type>inline void Rin(RG Type &x){
x=;RG int c=getchar();RG bool b=;
for(;c<||c>;c=getchar())
if(c==)b=;
for(;c>&&c<;c=getchar())
x=(x<<)+(x<<)+c-;
if(b)x=-x;
}
}; int n,_q,a[MaxN],_reflection[MaxN],block_size,block_cnt[MaxS][MaxN],belong[MaxN],lef[MaxS],rig[MaxS];
L block_rmq[MaxS][MaxS]; struct _pair{
int first,*second;
bool operator < (const _pair &other) const {
return first < other.first;
}
}c[MaxN]; inline L query(RG int x,RG int y){
static int tmp_tim[MaxN],tmp_cnt[MaxN],T=;
RG int _l=belong[x],_r=belong[y];
RG L res=block_rmq[_l+][_r-]; ++T;
if(_l == _r){
for(RG int i=x;i<=y;i++){
if(tmp_tim[a[i]] != T){
tmp_tim[a[i]]=T; tmp_cnt[a[i]]=;
}
++tmp_cnt[a[i]];
res=dmax(res,(L) _reflection[a[i]] * tmp_cnt[a[i]]);
}
return res;
}
for(RG int i=x;i<=:: rig[_l];i++){
if(tmp_tim[a[i]] != T){
tmp_tim[a[i]]=T; tmp_cnt[a[i]]=block_cnt[_r-][a[i]]-block_cnt[_l][a[i]];
}
++tmp_cnt[a[i]];
res=dmax(res,(L) _reflection[a[i]] * tmp_cnt[a[i]]);
}
for(RG int i=:: lef[_r];i<=y;i++){
if(tmp_tim[a[i]] != T){
tmp_tim[a[i]]=T; tmp_cnt[a[i]]=block_cnt[_r-][a[i]]-block_cnt[_l][a[i]];
}
++tmp_cnt[a[i]];
res=dmax(res,(L) _reflection[a[i]] * tmp_cnt[a[i]]);
}
return res;
} int main(){
io::Rin(n),io::Rin(_q);
block_size=static_cast<int>(sqrt(n)+1e-);
for(RG int i=;i<=n;i++)
io::Rin(c[i].first),c[i].second=&a[i];
std::sort(c+,c++n);
for(RG int i=,m=;i<=n;i++){
if(i== || c[i].first != c[i-].first)
_reflection[++m]=c[i].first;
*c[i].second=m;
}
for(RG int i=;i<=n;i++)
belong[i]=(i-)/block_size+;
for(RG int i=;i<=n;i++)
block_cnt[belong[i]][a[i]]++;
for(RG int i=;(i-)*block_size+<=n;i++)
lef[i]=(i-)*block_size+,rig[i]=dmin(i*block_size,n);
for(RG int i=;:: lef[i];i++)
for(RG int j=;j<=n;j++)
block_cnt[i][j]+=block_cnt[i-][j];
for(RG int i=;lef[i];i++){
static int tmp_cnt[MaxN]; RG L ans=0LL;
memset(tmp_cnt,,sizeof tmp_cnt);
for(RG int j=lef[i];j<=n;j++){
++tmp_cnt[a[j]];
ans=dmax(ans,(L) _reflection[a[j]] * tmp_cnt[a[j]]);
if(j == rig[belong[j]])
:: block_rmq[i][belong[j]]=ans;
}
}
while(_q--){
RG int x,y;
io::Rin(x),io::Rin(y);
printf("%lld\n",query(x,y));
}
fclose(stdin);
return ;
}

[bzoj4241][历史研究] (分块)的更多相关文章

  1. bzoj4241 历史研究——分块

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4241 就是分块,预处理出从第 i 块到 j 位置的答案,以及从第 i 块到最后位置间每个数出 ...

  2. [JOISC2014]歴史の研究/[BZOJ4241]历史研究

    [JOISC2014]歴史の研究/[BZOJ4241]历史研究 题目大意: 一个长度为\(n(n\le10^5)\)的数列\(A(A_i\le10^9)\),定义一个元素对一个区间\([l,r]\)的 ...

  3. 【题解】BZOJ4241: 历史研究(魔改莫队)

    [题解]BZOJ4241: 历史研究(魔改莫队) 真的是好题啊 题意 给你一个序列和很多组询问(可以离线),问你这个区间中\(\max\){元素出现个数\(\times\)元素权值} IOI国历史研究 ...

  4. 【BZOJ4241】历史研究 分块

    [BZOJ4241]历史研究 Description IOI国历史研究的第一人——JOI教授,最近获得了一份被认为是古代IOI国的住民写下的日记.JOI教授为了通过这份日记来研究古代IOI国的生活,开 ...

  5. BZOJ4241 历史研究

    Description IOI国历史研究的第一人——JOI教授,最近获得了一份被认为是古代IOI国的住民写下的日记.JOI教授为了通过这份日记来研究古代IOI国的生活,开始着手调查日记中记载的事件. ...

  6. BZOJ4241 历史研究 莫队 堆

    欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目 Description IOI国历史研究的第一人——JOI教授,最近获得了一份被认为是古代IOI国的住民写下的日记.JO ...

  7. BZOJ4241历史研究——回滚莫队

    题目描述 IOI国历史研究的第一人——JOI教授,最近获得了一份被认为是古代IOI国的住民写下的日记.JOI教授为了通过这份日记来研究古代IOI国的生活,开始着手调查日记中记载的事件. 日记中记录了连 ...

  8. BZOJ4241:历史研究(回滚莫队)

    Description IOI国历史研究的第一人——JOI教授,最近获得了一份被认为是古代IOI国的住民写下的日记.JOI教授为了通过这份日记来研究古代IOI国的生活,开始着手调查日记中记载的事件. ...

  9. BZOJ4241 历史研究(莫队)

    如果分块的话与区间众数没有本质区别.这里考虑莫队. 显然莫队时的删除可以用堆维护,但多了一个log不太跑得过. 有一种叫回滚莫队的trick,可以将问题变为只有加入操作.按莫队时分的块依次处理,一块中 ...

随机推荐

  1. bzoj4237 稻草人——分治

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4237 分治: 先把所有点按 y 排序,然后二分递归: 对于每个 mid ,计算经过它的矩形的 ...

  2. libnids 显示UDP数据报,编译,运行,正确。

    #include<stdio.h>#include<nids.h>#include<string.h>#include <sys/socket.h>#i ...

  3. [App Store Connect帮助]六、测试 Beta 版本(4.1) 管理 Beta 版构建版本:为构建版本添加测试员

    在“TestFlight”部分中,您可以查看您所有 App 版本的构建版本,并深入查看构建版本的详细信息.您也可以为某个构建版本添加群组或独立测试员. 必要职能:“帐户持有人”职能.“管理”职能或“A ...

  4. Django day 38 结算中心,支付中心,计算价格方法

    一:结算中心 二:支付中心 三:计算价格方法

  5. mysql的大数据量的查询

    mysql的大数据量查询分页应该用where 条件进行分页,limit 100000,100,mysql先查询100100数据量,查询完以后,将 这些100000数据量屏蔽去掉,用100的量,但是如果 ...

  6. 一个包含所有C++头文件的头函数

    #include<bits/stdc++.h> using namespace std; 使用方法和平常的头文件一样,#include<bits/stdc++.h>包含以下头文 ...

  7. 列表框、分组列表框、标签(label)、分组框(fieldset)、框架(frameset)

    列表框(select) 下拉列表,用户可以从一些可选项中选择. 示例:简单的下拉列表 <select name="country"> <option value= ...

  8. NPOI 导出excel数据超65535自动分表

    工作上遇到的问题,网上找了一些资料 整理了一个比较可行的解决方案. NPOI 大数据量分多个sheet导出 代码段 /// <summary> /// DataTable转换成Excel文 ...

  9. SqlServer学习-常用的sql语句-持续更新中

    1.获取数据库下的所有表名 select TABLE_NAME from information_schema.tables where TABLE_TYPE='Base TABLE' 2.随机取出1 ...

  10. Angular——依赖注入

    基本介绍 1.AngularJS采用模块化的方式组织代码,将一些通用逻辑封装成一个对象或函数,实现最大程度的复用,这导致了使用者和被使用者之间存在依赖关系. 2.所谓依赖注入是指在运行时自动查找依赖关 ...