[BZOJ 1926] 粟粟的书架
BZOJ的sillyB评测机各种无故CE,只好去Luogu上A了o(╯□╰)o
Solution:
从数据范围可以发现,这其实是2道题:
(1)一个$R*C$的矩形,每次询问一个子矩形的结果,$R,C\le 200$,$A[i][j]\le 1000$。
(2)一个$C$个数的序列,每次询问一个区间的结果,$C\le 5e5$。
1、前缀和+二分答案
找到此题的数据特点:$A[i][j]$极小,只有1000,使得$O(RC*1000)$的算法可行
于是我们维护1~i,1~j的矩形中大于k的数的个数及它们的总和,从而O(1)得出目标矩形的结果
$cnt[i][j][k]$表示行号在$[1,i]$,列号在$[1,j]$,厚度大于等于$k$的书的数目。
$sum[i][j][k]$表示行号在$[1,i]$,列号在$[1,j]$,厚度大于等于$k$的书的厚度之和。
每一次询问二分最小厚度(最小厚度是否可行具有单调性),求出值$t$,表示满足条件的情况下,最大的最小厚度。
然而由于有相同厚度,因此还要计算出厚度为$t$的书有多少本没有用上。
2、主席树
主席树解决的一类经典问题是求区间第K大的数
其实在这种背景下主席树的应用和线段树的性质关系并不大,可以理解为带了前缀和的二分查找树
由于其处理的区间具有对应关系,于是维护前缀的线段树间是可以相加减的,从而得到当前区间的信息
此题中厚度仅为1000,于是在每个点维护:
$cnt$:该前缀版本中,这个节点对应厚度区间内的书的数量。
$sum$:该前缀版本中,这个节点对应厚度区间内的书的厚度之和。
这样对答案在$root[end]-root[start-1]$的线段树中类似二分查找树地确定是进入左子树还是右子树即可
Code:
#include <bits/stdc++.h> using namespace std; inline int read()
{
char ch;int num,f=0;
while(!isdigit(ch=getchar())) f|=(ch=='-');
num=ch-'0';
while(isdigit(ch=getchar())) num=num*10+ch-'0';
return f?-num:num;
} const int MAXN=205,MAXM=1005;
const int N=5e5+10,M=1e7+10; struct FunSeg
{
int ls,rs,sum,cnt;
}seg[M];
int root[N],S[N],cnt; int n,m,q,dat[MAXN][MAXN],f[MAXN][MAXN][MAXM],g[MAXN][MAXN][MAXM]; int cal_f(int x1,int y1,int x2,int y2,int k)
{
return f[x2][y2][k]-f[x1-1][y2][k]-f[x2][y1-1][k]+f[x1-1][y1-1][k];
} int cal_g(int x1,int y1,int x2,int y2,int k)
{
return g[x2][y2][k]-g[x1-1][y2][k]-g[x2][y1-1][k]+g[x1-1][y1-1][k];
} void solve1()
{
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++) dat[i][j]=read(); for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
{
for(int k=1;k<=1000;k++)
f[i][j][k]=f[i-1][j][k]+f[i][j-1][k]-f[i-1][j-1][k],
g[i][j][k]=g[i-1][j][k]+g[i][j-1][k]-g[i-1][j-1][k];
f[i][j][dat[i][j]]+=dat[i][j];g[i][j][dat[i][j]]++;
} for(int k=999;k>=1;k--)
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
f[i][j][k]+=f[i][j][k+1],g[i][j][k]+=g[i][j][k+1]; while(q--)
{
int x1=read(),y1=read(),x2=read(),y2=read(),h=read(),l=1,r=1000; while(l<=r)
{
int mid=(l+r)>>1;
if(cal_f(x1,y1,x2,y2,mid)>=h) l=mid+1;
else r=mid-1;
} if(!r){puts("Poor QLW");continue;};
cout << cal_g(x1,y1,x2,y2,r)-(cal_f(x1,y1,x2,y2,r)-h)/r << endl; //将这个厚度中多余的书减掉
}
} void Update(int x,int &y,int val,int l,int r)
{
y=++cnt;seg[y]=seg[x]; //不用改变的节点沿用上一次已经完全构造完的
seg[y].sum+=val;seg[y].cnt++;
if(l==r) return; int mid=(l+r)>>1;
if(val<=mid) Update(seg[x].ls,seg[y].ls,val,l,mid);
else Update(seg[x].rs,seg[y].rs,val,mid+1,r);
} int Query(int x,int y,int val,int l,int r)
{
if(l==r) return val/l+(val%l>0); //对整除情况特殊处理 int mid=(l+r)>>1;
int t=seg[seg[y].rs].sum-seg[seg[x].rs].sum;
if(t>=val) return Query(seg[x].rs,seg[y].rs,val,mid+1,r); //厚度范围是(mid+1,r)时足够
else return Query(seg[x].ls,seg[y].ls,val-t,l,mid)+seg[seg[y].rs].cnt-seg[seg[x].rs].cnt; //不够时
} void solve2()
{
for(int i=1;i<=m;i++)
S[i]=read(),Update(root[i-1],root[i],S[i],1,1000),S[i]+=S[i-1]; //以1~i-1的前缀树为模板建树 while(q--)
{
int x1=read(),y1=read(),x2=read(),y2=read(),h=read();
if(S[y2]-S[y1-1]<h) puts("Poor QLW");
else cout << Query(root[y1-1],root[y2],h,1,1000) << endl;
}
} int main()
{
n=read();m=read();q=read();
if(n>1) solve1();
else solve2();
return 0;
}
1、注意数据范围的特点
如每个点数值的范围不大,可以考虑将 大于等于k的个数/总和 这样与数值范围相关的条件加入递推式中
2、如果 一个量是否符合要求 具有单调性,可以对这个量二分答案
3、主席树:
大多数时候是离线的数据结构
需要知道数据的范围。每个节点维护的是1~i个数中处于数据范围[L,R]的数的信息。
也就是说每棵前缀树中维护的是数据范围[1,1000]的区间信息
4、注意x组成y至少要多少个这类问题的书写:
y=y/x+(y%x>0) || y=(y+x-1)/x
[BZOJ 1926] 粟粟的书架的更多相关文章
- bzoj 1926: [Sdoi2010]粟粟的书架 (主席树+二分)
链接:https://www.lydsy.com/JudgeOnline/problem.php?id=1926 题面; 1926: [Sdoi2010]粟粟的书架 Time Limit: 30 Se ...
- Bzoj 1926: [Sdoi2010]粟粟的书架(二分答案+乱搞+主席树)
1926: [Sdoi2010]粟粟的书架 Time Limit: 30 Sec Memory Limit: 552 MB Description 幸福幼儿园 B29 班的粟粟是一个聪明机灵.乖巧可爱 ...
- 【刷题】BZOJ 1926 [Sdoi2010]粟粟的书架
Description 幸福幼儿园 B29 班的粟粟是一个聪明机灵.乖巧可爱的小朋友,她的爱好是画画和读书,尤其喜欢 Thomas H. Cormen 的文章.粟粟家中有一个 R行C 列的巨型书架,书 ...
- BZOJ 1926: [Sdoi2010]粟粟的书架(主席树,二分答案)
BZOJ 1926: [Sdoi2010]粟粟的书架(主席树,二分答案) 题意 : 给你一个长为\(R\)宽为\(C\)的矩阵,第\(i\)行\(j\)列的数为\(P_{i,j}\). 有\(m\)次 ...
- bzoj 1926: [Sdoi2010]粟粟的书架
#include<cstdio> #include<iostream> #define N 201 #define M 500008 using namespace std; ...
- BZOJ.1926.[SDOI2010]粟粟的书架(前缀和 主席树 二分)
题目链接 题意: 在给定矩形区域内找出最少的数,满足和>=k.输出数的个数.两种数据范围. 0~50 注意到(真没注意到...)P[i,j]<=1000,我们可以利用前缀和预处理. num ...
- 粟粟的书架(bzoj 1926)
Description 幸福幼儿园 B29 班的粟粟是一个聪明机灵.乖巧可爱的小朋友,她的爱好是画画和读书,尤其喜欢 Thomas H. Co rmen 的文章.粟粟家中有一个 R行C 列的巨型书架, ...
- 【BZOJ-1926】粟粟的书架 二分 + 前缀和 + 主席树
1926: [Sdoi2010]粟粟的书架 Time Limit: 30 Sec Memory Limit: 552 MBSubmit: 616 Solved: 238[Submit][Statu ...
- [BZOJ1926][SDOI2010]粟粟的书架
BZOJ Luogu Description 幸福幼儿园 B29 班的粟粟是一个聪明机灵.乖巧可爱的小朋友,她的爱好是画画和读书,尤其喜欢 Thomas H. Cormen 的文章.粟粟家中有一个 R ...
- bzoj1926[Sdoi2010]粟粟的书架 二分 主席树
1926: [Sdoi2010]粟粟的书架 Time Limit: 30 Sec Memory Limit: 552 MBSubmit: 1064 Solved: 421[Submit][Stat ...
随机推荐
- [HEOI2015]最短不公共子串
四合一的题. 简单粗暴的方法: 子串匹配——SAM 子序列匹配——序列自动机 关于序列自动机:序列自动机—— [FJOI2016]所有公共子序列问题 (其实这个玩意没有什么,n+1个点,每个点的字符集 ...
- 【bzoj2141】排队 [国家集训队2011]排队(魏铭) 树套树 线段树套替罪羊树
这个题就是动态偏序对,每次操作做两个删除两个插入就好了. #include<cstdio> #include<iostream> #include<cstring> ...
- shell编程 if 注意事项
read -n 1 -p "Let's go(y or n):" if [ "$REPLY"x = "y"x -o "$REPLY ...
- maven项目在eclipse tomcat正常运行
转摘自:http://binary.duapp.com/2013/10/1577.html 配置下部署路径即可.
- 设计模式开篇综述(Java)
设计原则是规范,设计模式是技巧.如果在项目中能够灵活运用这些基础知识,那么我相信一定会得到意想不到的收获. 接下来的时间里,我将继续学习设计模式,将对每一个设计模式从以下几点进行分析和学习,如有不妥当 ...
- phpAdmin 修改密码后拒绝登陆
phpMyadmin没配置正确,打开 phpMyadmin 目录找到config.inc.php文件,查找到$cfg['Servers'][$i]['password']='';这行,在''中输入你正 ...
- vivo面试学习1(io和nio)
一.io流(一次从open到底层的操作) 输入和输出流 IO流 字节流 Reader.Writer 字符流 InputStream.OutputStream 字节流:可以处理所有bit为单位存储的文件 ...
- USACO月赛2005 january volume
2013-09-18 08:12 由题可知,ans=∑i ∑j(x[i]-x[j]) 最后整理完之后应该是不同系数的X[i]相加,所以这道题就成了求不同x[i]的系数 对于X[i],它需要减前面(i ...
- [转]Native进程的运行过程
Native进程的运行过程 一般程序的启动步骤,可以用下图描述.程序由内核加载分析,使用linker链接需要的共享库,然后从c运行库的入口开始执行. 通常,native进程是由shell或者init启 ...
- linux知识复习1-dup dup2
#include <sys/stat.h> #include <string.h> #include <fcntl.h> #include <stdio.h& ...