\(\color{#0066ff}{ 题目描述 }\)

给你一个N*N的矩阵,不用算矩阵乘法,但是每次询问一个子矩形的第K小数。

\(\color{#0066ff}{输入格式}\)

第一行两个数N,Q,表示矩阵大小和询问组数;

接下来N行N列一共N*N个数,表示这个矩阵;

再接下来Q行每行5个数描述一个询问:x1,y1,x2,y2,k表示找到以(x1,y1)为左上角、以(x2,y2)为右下角的子矩形中的第K小数。

\(\color{#0066ff}{输出格式}\)

对于每组询问输出第K小的数。

\(\color{#0066ff}{输入样例}\)

2 2
2 1
3 4
1 2 1 2 1
1 1 2 2 3

\(\color{#0066ff}{输出样例}\)

1
3

\(\color{#0066ff}{数据范围与提示}\)

矩阵中数字是10^9以内的非负整数;

20%的数据:N<=100,Q<=1000;

40%的数据:N<=300,Q<=10000;

60%的数据:N<=400,Q<=30000;

100%的数据:N<=500,Q<=60000。

\(\color{#0066ff}{题解}\)

整体二分, 显然一看n的范围,直接用二维树状数组来维护这个东西即可

// luogu-judger-enable-o2
#include<bits/stdc++.h>
#define LL long long
LL in() {
char ch; LL x = 0, f = 1;
while(!isdigit(ch = getchar()))(ch == '-') && (f = -f);
for(x = ch ^ 48; isdigit(ch = getchar()); x = (x << 1) + (x << 3) + (ch ^ 48));
return x * f;
}
const int maxn = 4e5 + 10;
struct Tree {
protected:
int n;
int st[555][555];
int low(int x) { return x & (-x); }
int getans(int a, int b) {
int re = 0;
for(int i = a; i; i -= low(i))
for(int j = b; j; j -= low(j))
re += st[i][j];
return re;
}
public:
void resize(int len) { n = len; }
void add(int x, int y, int k) {
for(int i = x; i <= n; i += low(i))
for(int j = y; j <= n; j += low(j))
st[i][j] += k;
}
int query(int a, int b, int x, int y) {
return getans(x, y) - getans(x, b - 1) - getans(a - 1, y) + getans(a - 1, b - 1);
}
}s;
struct node {
int a, b, x, y, k, id;
node(int a = 0, int b = 0, int x = 0, int y = 0, int k = 0, int id = 0): a(a), b(b), x(x), y(y), k(k), id(id) {}
}q[maxn], ql[maxn], qr[maxn];
int ans[maxn], n, m, num;
void work(int l, int r, int nl, int nr) {
if(l > r || nl > nr) return;
if(l == r) {
for(int i = nl; i <= nr; i++) if(q[i].id) ans[q[i].id] = l;
return;
}
int mid = (l + r) >> 1, cntl = 0, cntr = 0;
for(int i = nl; i <= nr; i++) {
if(q[i].id) {
int k = s.query(q[i].a, q[i].b, q[i].x, q[i].y);
if(q[i].k <= k) ql[++cntl] = q[i];
else q[i].k -= k, qr[++cntr] = q[i];
}
else {
if(q[i].k <= mid) s.add(q[i].a, q[i].b, 1), ql[++cntl] = q[i];
else qr[++cntr] = q[i];
}
}
for(int i = nl; i <= nr; i++) if(!q[i].id && q[i].k <= mid) s.add(q[i].a, q[i].b, -1);
for(int i = 1; i <= cntl; i++) q[nl + i - 1] = ql[i];
for(int i = 1; i <= cntr; i++) q[nl + cntl + i - 1] = qr[i];
work(l, mid, nl, nl + cntl - 1), work(mid + 1, r, nl + cntl, nr);
}
int main() {
n = in(), m = in();
s.resize(n);
for(int i = 1; i <= n; i++)
for(int j = 1; j <= n; j++)
q[++num] = node(i, j, 0, 0, in(), 0);
for(int i = 1; i <= m; i++) {
num++;
q[num].a = in(), q[num].b = in();
q[num].x = in(), q[num].y = in();
q[num].k = in(), q[num].id = i;
}
work(0, 1e9, 1, num);
for(int i = 1; i <= m; i++) printf("%d\n", ans[i]);
return 0;
}

P1527 [国家集训队]矩阵乘法的更多相关文章

  1. 洛谷 P1527 [国家集训队]矩阵乘法 解题报告

    P1527 [国家集训队]矩阵乘法 题目描述 给你一个\(N*N\)的矩阵,不用算矩阵乘法,但是每次询问一个子矩形的第\(K\)小数. 输入输出格式 输入格式: 第一行两个数\(N,Q\),表示矩阵大 ...

  2. [洛谷P1527] [国家集训队]矩阵乘法

    洛谷题目链接:[国家集训队]矩阵乘法 题目背景 原 <补丁VS错误>请前往P2761 题目描述 给你一个N*N的矩阵,不用算矩阵乘法,但是每次询问一个子矩形的第K小数. 输入输出格式 输入 ...

  3. 洛谷P1527 [国家集训队] 矩阵乘法 [整体二分,二维树状数组]

    题目传送门 矩阵乘法 题目描述 给你一个N*N的矩阵,不用算矩阵乘法,但是每次询问一个子矩形的第K小数. 输入输出格式 输入格式: 第一行两个数N,Q,表示矩阵大小和询问组数: 接下来N行N列一共N* ...

  4. 洛谷$P1527$ [国家集训队]矩阵乘法 整体二分

    正解:整体二分 解题报告: 传送门$QwQ$ 阿看到这种查询若干次第$k$小显然就想到整体二分$QwQ$? 然后现在就只要考虑怎么快速求出一个矩形内所有小于某个数的数的个数? 开始我的想法是离散化然后 ...

  5. P1527 [国家集训队]矩阵乘法 [整体二分]

    权值排序,整体二分,没了. // by Isaunoya #include <bits/stdc++.h> using namespace std; #define rep(i, x, y ...

  6. P1527 [国家集训队]矩阵乘法(整体二分)

    Link 整体二分的经典例题. 对于整体二分,我个人的理解是二分答案套分治. 具体来说就是对答案进行二分,然后对于询问进行类似于权值线段树求区间第 \(k\) 大的分治做法. 首先,我们暴力做法就是对 ...

  7. 【LG1527】[国家集训队]矩阵乘法

    [LG1527][国家集训队]矩阵乘法 题面 洛谷 题解 我也不知道为什么取个这样的名字... 其实就是区间\(kth\)扩展到二维 还是用整体二分搞啦,把树状数组换成二维的 其他的基本没有什么差别 ...

  8. Luogu-1527 [国家集训队]矩阵乘法

    Luogu-1527 [国家集训队]矩阵乘法 题面 Luogu-1527 题解 昨天学CDQ分治时做了一些题,但是因为题(wo)太(tai)水(lan)了(le)并没有整理 学了一晚上的整体二分,拿这 ...

  9. [Luogu1527][国家集训队]矩阵乘法

    luogu 题意 给你一个\(N*N\)的矩阵,每次询问一个子矩形的第K小数.(居然连修改都不带的) \(N\le500,Q\le60000\) sol 整体二分+二维树状数组裸题. 复杂度是\(O( ...

随机推荐

  1. js生成邀请码(2)

    //生成邀请码方法一 /*function createInviteCode() { var s = [],a=6,b=10; var chars = "123456789QWERTYUIP ...

  2. 2015.12.14 MDI(多文档窗口结构)设置基本解决,折腾一天,部分解决存在已久的问题。但效果仍不如临时航线的MDI窗体结构。

    创建从一个窗口弹出多个子窗口的结构叫MDI窗体结构 如果不按MDI结构管理,最简单的做法是: 在窗体A上添加菜单或按钮,在菜单或按钮事件中添加弹出B窗体代码: B b = new B(); b.sho ...

  3. apache server和tomcat集群配置三:水平集群下的tomcat集群配置

    在jsp文件中加入以下代码,用来测试是否共享session: SessionID: <%= session.getId() %> 之前尝试在linux中,但是因为模拟环境是虚拟机,虚拟机只 ...

  4. python爬虫(10)--PyQuery的用法

    简介 pyquery 可让你用 jQuery 的语法来对 xml 进行操作.这I和 jQuery 十分类似.如果利用 lxml,pyquery 对 xml 和 html 的处理将更快. 初始化 在这里 ...

  5. oracle 之sys 、system区别

    角色 1)最重要的区别,存储的数据的重要性不同sys--所有oracle的数据字典的基表和视图都存放在sys用户中,这些基表和视图对于oracle的运行是至关重要的,由数据库自己维护,任何用户都不能手 ...

  6. 第3章 ZooKeeper基本数据模型 3-1 zk数据模型介绍

    基本数据模型是zookeeper的重点. 它是参照Linux/Unix的目录结构. 子节点就相当于是父目录下的一个子目录,在zookeeper里面它是称之为节点,父节点和子节点,然后每一个节点就会有一 ...

  7. JavaScript基础笔记集合(转)

    JavaScript基础笔记集合   JavaScript基础笔记集合   js简介 js是脚本语言.浏览器是逐行的读取代码,而传统编程会在执行前进行编译   js存放的位置 html脚本必须放在&l ...

  8. ubuntu16.04 安装opencv3.4

    1.去官网下载opencv,在本教程中选用的时opencv3.4.1,其他版本的配置方法异曲同工. 下载链接http://opencv.org/releases.html,选择sources版本 2. ...

  9. (转)Linux网络协议栈(三)——网络设备(1)

    网络设备(network device)是内核对网络适配器(硬件)的抽象与封装,并为各个协议实例提供统一的接口,它是硬件与内核的接口,它有两个特征:(1)    作为基于硬件的网络适配器与基于软件的协 ...

  10. JavaWeb_增强for循环

    引入增强for循环的原因:在JDK5以前的版本中,遍历数组或集合中的元素,需要先获得数组的长度或集合的迭代器,比较麻烦. JDK5中定义了一种新的语法----增强for循环,以简化此类操作.增强for ...