bzoj1926[sdoi2010]粟粟的书架主席树+二分+前缀和

Mychael Mychael     2022-10-24     685

关键词:

题目

幸福幼儿园 B29 班的粟粟是一个聪明机灵、乖巧可爱的小朋友,她的爱好是画画和读书,尤其喜欢 Thomas H. Co
rmen 的文章。粟粟家中有一个 R行C 列的巨型书架,书架的每一个位置都摆有一本书,上数第i 行、左数第j 列
摆放的书有Pi,j页厚。粟粟每天除了读书之外,还有一件必不可少的工作就是摘苹果,她每天必须摘取一个指定的
苹果。粟粟家果树上的苹果有的高、有的低,但无论如何凭粟粟自己的个头都难以摘到。不过她发现, 如果在脚
下放上几本书,就可以够着苹果;她同时注意到,对于第 i 天指定的那个苹果,只要她脚下放置书的总页数之和
不低于Hi,就一定能够摘到。由于书架内的书过多,父母担心粟粟一天内就把所有书看完而耽误了上幼儿园,于是
每天只允许粟粟在一个特定区域内拿书。这个区域是一个矩形,第 i 天给定区域的左上角是上数第 x1i行的左数
第 y1i本书,右下角是上数第 x2i行的左数第y2i本书。换句话说,粟粟在这一天,只能在这﹙x2i-x1i+1﹚×﹙
y2i-y1i+1﹚本书中挑选若干本垫在脚下,摘取苹果。粟粟每次取书时都能及时放回原位,并且她的书架不会再
撤下书目或换上新书,摘苹果的任务会一直持续 M天。给出每本书籍的页数和每天的区域限制及采摘要求,请你告
诉粟粟,她每天至少拿取多少本书,就可以摘到当天指定的苹果。

输入格式

第一行是三个正整数R,C,M。
接下来是一个R行C列的矩阵,从上到下、从左向右依次给出了每本书的页数Pi,j。
接下来M行,第i行给出正整数x1i,y1i,x2i,y2i,Hi,表示第i天的指定区域是﹙x1i,y1i﹚与﹙x2i,y2i﹚间
的矩形,总页数之和要求不低于Hi。
保证1≤x1i≤x2i≤R,1≤y1i≤y2i≤C。

输出格式

有M行,第i 行回答粟粟在第 i 天时为摘到苹果至少需要 拿取多少本书。如果即使取走所有书都无法摘到苹果,
则在该行输出“Poor QLW” (不含引号)。

输入样例

5 5 7

14 15 9 26 53

58 9 7 9 32

38 46 26 43 38

32 7 9 50 28

8 41 9 7 17

1 2 5 3 139

3 1 5 5 399

3 3 4 5 91

4 1 4 1 33

1 3 5 4 185

3 3 4 3 23

3 1 3 3 108

输出样例

6

15

2

Poor QLW

9

1

3

提示

对于 10%的数据,满足 R, C≤10;

对于 20%的数据,满足 R, C≤40;

对于 50%的数据,满足 R, C≤200,M≤200,000;

另有 50%的数据,满足 R=1,C≤500,000,M≤20,000;

对于 100%的数据,满足 1≤Pi,j≤1,000,1≤Hi≤2,000,000,000

题解

此题二合一
我数据结构学傻了,二维写了一个树状数组套主席树,然后T了。。。

对于一条链,二分答案 + 主席树判定
对于二维,开一个数组num[x][y][k]表示(1,1)到(x,y)中所有>=k的数的总和,tot[x][y][k]表示有多少这样的数
然后也可以二分答案

#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<algorithm>
#define LL long long int
#define Redge(u) for (int k = h[u],to; k; k = ed[k].nxt)
#define REP(i,n) for (int i = 1; i <= (n); i++)
#define BUG(s,n) for (int i = 1; i <= (n); i++) cout<<s[i]<<‘ ‘; puts("");
#define lbt(x) (x & -x)
using namespace std;
const int maxn = 500005,maxm = 10000005,INF = 1000000000;
inline int read()
    int out = 0,flag = 1; char c = getchar();
    while (c < 48 || c > 57)if (c == ‘-‘) flag = -1; c = getchar();
    while (c >= 48 && c <= 57)out = (out << 3) + (out << 1) + c - 48; c = getchar();
    return out * flag;

int ls[maxm],rs[maxm],sum[maxm],cnt[maxm],rt[maxn],siz;
int n,m,Q,N = 1000;
void modify(int& u,int pre,int l,int r,int pos)
    u = ++siz; ls[u] = ls[pre]; rs[u] = rs[pre];
    cnt[u] = cnt[pre] + 1; sum[u] = sum[pre] + pos;
    if (l == r) return;
    int mid = l + r >> 1;
    if (mid >= pos) modify(ls[u],ls[pre],l,mid,pos);
    else modify(rs[u],rs[pre],mid + 1,r,pos);

int query(int u,int v,int l,int r,int k)
    if (cnt[u] - cnt[v] == k) return sum[u] - sum[v];
    if (l == r) return (sum[u] - sum[v]) / (cnt[u] - cnt[v]) * k;
    int mid = l + r >> 1,t = cnt[rs[u]] - cnt[rs[v]];
    if (t < k) return query(ls[u],ls[v],l,mid,k - t) + sum[rs[u]] - sum[rs[v]];
    else return query(rs[u],rs[v],mid + 1,r,k);

void solve1()
    for (int i = 1; i <= m; i++)
        modify(rt[i],rt[i - 1],1,N,read());
    int L,R,h;
    while (Q--)
        read(); L = read() - 1; read(); R = read(); h = read();
        if (sum[rt[R]] - sum[rt[L]] < h)
            puts("Poor QLW"); continue;
        
        int l = 1,r = cnt[rt[R]] - cnt[rt[L]],mid;
        while (l < r)
            mid = l + r >> 1;
            if (query(rt[R],rt[L],1,N,mid) >= h) r = mid;
            else l = mid + 1;
        
        printf("%d\n",l);
    

int num[205][205][1005],tot[205][205][1005],x,y,xx,yy,h;
int S(int mid)
    return num[xx][yy][mid] - num[x - 1][yy][mid] - num[xx][y - 1][mid] + num[x - 1][y - 1][mid];

int C(int mid)
    return tot[xx][yy][mid] - tot[x - 1][yy][mid] - tot[xx][y - 1][mid] + tot[x - 1][y - 1][mid];

void solve2()
    for (int i = 1; i <= n; i++)
        for (int j = 1; j <= m; j++)
            int x = read();
            for (int k = x; k; k--)
                num[i][j][k] = x,tot[i][j][k] = 1;
            for (int k = 1; k <= N; k++)
                num[i][j][k] += num[i - 1][j][k] + num[i][j - 1][k] - num[i - 1][j - 1][k],
                tot[i][j][k] += tot[i - 1][j][k] + tot[i][j - 1][k] - tot[i - 1][j - 1][k];
        
    int l,r,mid;
    while (Q--)
        x = read(); y = read(); xx = read(); yy = read(); h = read();
        if (S(1) < h) puts("Poor QLW"); continue;
        l = 1; r = 1000;
        while (l < r)
            mid = l + r + 1 >> 1;
            if (S(mid) >= h) l = mid;
            else r = mid - 1;
        
        int t = h - S(l + 1);
        printf("%d\n",C(l + 1) + (t % l == 0 ? t / l : t / l + 1));
    

int main()
    n = read(); m = read(); Q = read();
    if (n == 1) solve1();
    else solve2();
    return 0;

ac日记——[sdoi2010]粟粟的书架bzoj1926

1926 思路:  主席树+二分水题; 代码:#include<bits/stdc++.h>usingnamespacestd;#definemaxn500005#definemaxr205#definemaxn_maxn*13#defineFalseAns"PoorQLW"intn,m,q,lc[maxn_],rc[maxn_],num[maxn_],ci[max 查看详情

bzoj1926:[sdoi2010]粟粟的书架

1926:[Sdoi2010]粟粟的书架TimeLimit:30Sec  MemoryLimit:552MBSubmit:1013  Solved:396[Submit][Status][Discuss]Description幸福幼儿园B29班的粟粟是一个聪明机灵、乖巧可爱的小朋友,她的爱好是画画和读书,尤其喜欢ThomasH.Cormen的文章。粟... 查看详情

bzoj1926[sdoi2010]粟粟的书架

题解:这是两道题前50%:发现p[i][j]很小,于是记录f[i][j][k]表示(1,1)~(i,j)这个子矩阵内>=k的书的总高度,g[i][j][k]记录本数查询是二分答案就好了后50%:主席树,右子树够了就向右走,否则向左走 #include<iostream>#includ... 查看详情

bzoj1926[sdoi2010]粟粟的书架

 Description幸福幼儿园B29班的粟粟是一个聪明机灵、乖巧可爱的小朋友,她的爱好是画画和读书,尤其喜欢ThomasH.Cormen的文章。粟粟家中有一个R行C列的巨型书架,书架的每一个位置都摆有一本书,上数第i行、左数第j列摆放的... 查看详情

刷题bzoj1926[sdoi2010]粟粟的书架

Description幸福幼儿园B29班的粟粟是一个聪明机灵、乖巧可爱的小朋友,她的爱好是画画和读书,尤其喜欢ThomasH.Cormen的文章。粟粟家中有一个R行C列的巨型书架,书架的每一个位置都摆有一本书,上数第i行、左数第j列摆放的书有P... 查看详情

1926:[sdoi2010]粟粟的书架

1926:[Sdoi2010]粟粟的书架TimeLimit: 30Sec  MemoryLimit: 552MBSubmit: 807  Solved: 321[Submit][Status][Discuss]Description幸福幼儿园B29班的粟粟是一个聪明机灵、乖巧可爱的小朋友,她的爱好是画画和读书,尤其喜欢 查看详情

[bzoj1926]粟粟的书架(代码片段)

BZOJ:https://www.lydsy.com/JudgeOnline/problem.php?id=1926Luogu:https://www.luogu.org/problemnew/show/P2468BZOJ的sillyB评测机各种无故CE,只好去Luogu上A了o(╯□╰)o Algorithm:从数据范围可以发现,这其实是2道题:1、一个R×C的矩形,每次询问一个子矩... 查看详情

[sdoi2010]粟粟的书架[主席树](代码片段)

[SDOI2010]粟粟的书架考虑暴力怎么做显然是提取出来(x2-x1+1)*(y2-y1+1)个数字拿出来然后从大到小排序然后就可以按次取数了…然而接下来看数据范围(50\%r,cleq200)(50\%r=1,cleq5*10^5)值域(in[1,1000])对于前50%可以用个前缀和搞定…令(sum_i,j,k)... 查看详情

[sdoi2010]粟粟的书架(代码片段)

Description幸福幼儿园B29班的粟粟是一个聪明机灵、乖巧可爱的小朋友,她的爱好是画画和读书,尤其喜欢ThomasH.Cormen的文章。粟粟家中有一个R行C列的巨型书架,书架的每一个位置都摆有一本书,上数第i行、左数第j列摆放的书有P... 查看详情

sdoi2010粟粟的书架(代码片段)

传送门代码1#include<iostream>2#include<cstring>3#include<cstdio>4#include<algorithm>5#include<math.h>6#definemaxn5000057usingnamespacestd;8intR,C,M;9intcont[1005];10inthe[205] 查看详情

[sdoi2010]粟粟的书架

题目大意:网址:https://daniu.luogu.org/problemnew/show/2468大意:本题有两问:[1]给定一个(R*C)的带权矩阵,询问(2×10^5)次在一个子矩阵内至少选择多少个点使他们的权值和大于(H)。[2]给定长度为L的一条链,询问(2×10^4)次在一个区间([L,... 查看详情

[sdoi2010]粟粟的书架(代码片段)

[SDOI2010]粟粟的书架题意:一个R*C的矩阵,每个位置都有个数page[ij],现在选定一个小矩阵范围(给左上角坐标,和右下角坐标),问这个范围内的数总和是否大于h,如果大于h的话最少选几个数aij对于50%的数... 查看详情

sdoi2010粟粟的书架lg2468(可持久化,前缀和)(代码片段)

题面见https://www.luogu.org/problemnew/show/P2468然后这道题属于合二为一题,看一眼数据范围就能发现首先我们先考虑50分,二维前缀和维护一下(反正我不记得公式,手推了半天)tot[i][j][k]表示矩阵(1,1)到(i,j)中数值大等于k的总... 查看详情

粟粟的书架题解(代码片段)

粟粟的书架题解第一次见到这种二合一的题,开始的时候居然死磕二维主席树,又是屈辱看题解系列,其实很比较好做第一部分(R,C≤200,M≤200000,1≤Pi,j≤1,000)这一部分可以用两个数组来记录:(num[i][j][k]):代表1~i,1~j的矩形中小于... 查看详情

可持久化汇总(讲解+题目)

...OI2016]神秘数CF484ESignonFenceCF1080FKatyaandSegmentsSetsP2468[SDOI2010]粟粟的书架P3302[SD 查看详情

bzoj1878:[sdoi2009]hh的项链(主席树)

题目传送门:QWQ 分析莫队也能做,但我想练练主席树。求k-th一样维护第i个时候的线段树,线段树来维护区间不同数。然后查询时可以通过上下界小优化一波。但是我的代码丑陋无比,常数巨大(捂脸  代码#include<b... 查看详情

bzoj3123[sdoi2013]森林主席树启发式合并

我们直接按父子关系建主席树,然后记录倍增方便以后求LCA,同时用并查集维护根节点,而且还要记录根节点对应的size,用来对其启发式合并,然后每当我们合并的时候我们都要暴力拆小的一部分重复以上部分,总时间复杂度... 查看详情

bzoj3123sdoi2013森林

...,那么每次启发式合并就可以了。查询路径(k)小似乎需要主席树,那么把主席树和倍增表一起暴力重构就好了。  然后发现这样的空间复杂度是(O(nlog^2n))的。感觉非常不靠谱,于是滚去写了个节点回收站……然后发现主席树节... 查看详情