P1527 [国家集训队]矩阵乘法
\(\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 [国家集训队]矩阵乘法的更多相关文章
- 洛谷 P1527 [国家集训队]矩阵乘法 解题报告
P1527 [国家集训队]矩阵乘法 题目描述 给你一个\(N*N\)的矩阵,不用算矩阵乘法,但是每次询问一个子矩形的第\(K\)小数. 输入输出格式 输入格式: 第一行两个数\(N,Q\),表示矩阵大 ...
- [洛谷P1527] [国家集训队]矩阵乘法
洛谷题目链接:[国家集训队]矩阵乘法 题目背景 原 <补丁VS错误>请前往P2761 题目描述 给你一个N*N的矩阵,不用算矩阵乘法,但是每次询问一个子矩形的第K小数. 输入输出格式 输入 ...
- 洛谷P1527 [国家集训队] 矩阵乘法 [整体二分,二维树状数组]
题目传送门 矩阵乘法 题目描述 给你一个N*N的矩阵,不用算矩阵乘法,但是每次询问一个子矩形的第K小数. 输入输出格式 输入格式: 第一行两个数N,Q,表示矩阵大小和询问组数: 接下来N行N列一共N* ...
- 洛谷$P1527$ [国家集训队]矩阵乘法 整体二分
正解:整体二分 解题报告: 传送门$QwQ$ 阿看到这种查询若干次第$k$小显然就想到整体二分$QwQ$? 然后现在就只要考虑怎么快速求出一个矩形内所有小于某个数的数的个数? 开始我的想法是离散化然后 ...
- P1527 [国家集训队]矩阵乘法 [整体二分]
权值排序,整体二分,没了. // by Isaunoya #include <bits/stdc++.h> using namespace std; #define rep(i, x, y ...
- P1527 [国家集训队]矩阵乘法(整体二分)
Link 整体二分的经典例题. 对于整体二分,我个人的理解是二分答案套分治. 具体来说就是对答案进行二分,然后对于询问进行类似于权值线段树求区间第 \(k\) 大的分治做法. 首先,我们暴力做法就是对 ...
- 【LG1527】[国家集训队]矩阵乘法
[LG1527][国家集训队]矩阵乘法 题面 洛谷 题解 我也不知道为什么取个这样的名字... 其实就是区间\(kth\)扩展到二维 还是用整体二分搞啦,把树状数组换成二维的 其他的基本没有什么差别 ...
- Luogu-1527 [国家集训队]矩阵乘法
Luogu-1527 [国家集训队]矩阵乘法 题面 Luogu-1527 题解 昨天学CDQ分治时做了一些题,但是因为题(wo)太(tai)水(lan)了(le)并没有整理 学了一晚上的整体二分,拿这 ...
- [Luogu1527][国家集训队]矩阵乘法
luogu 题意 给你一个\(N*N\)的矩阵,每次询问一个子矩形的第K小数.(居然连修改都不带的) \(N\le500,Q\le60000\) sol 整体二分+二维树状数组裸题. 复杂度是\(O( ...
随机推荐
- Oracle 监听莫名死掉
有一台oracle 10g的监听莫名死掉,进行查看 select * from v$version Oracle Database 10g Enterprise Edition Release 10. ...
- idea debug的时候 启动起来超级慢
- 【转】href="#"与"javascript:void(0);"的区别
在工作中,如果我们想把a标签中的链接置成空链接,我们一般会用两种方法: 1 <a href="#" target="_blank"></a&g ...
- 部署和调优 2.4 tomcat安装
下载tamcet 官网 http://tomcat.apache.org/ 左侧选择版本 复制下载链接 切换到下载目录 cd /usr/local/src linux wget wget http:/ ...
- C语言学习笔记--#error 、 #line 和 #pragma 的使用
1. #error 的用法 (1)#error 是一种预编译器指示字,用于生成一个编译错误消息 (2)用法:#error message //注意:message 不需要用双引号包围 (3)#erro ...
- sql中IN的用法
1.和where配合使用 IN操作符允许我们在where的子句中规定多个值 SELECT * FROM Persons WHERE LastName IN ('Adams','Carter') 这句 ...
- 每天一道算法题(24)——自定义幂函数pow
double myPower(double base, int exponent){ if(exponent==0) return 1; if(exponent==1) return base; if ...
- JS继承方式详解
js继承的概念 js里常用的如下两种继承方式: 原型链继承(对象间的继承) 类式继承(构造函数间的继承) 由于js不像java那样是真正面向对象的语言,js是基于对象的,它没有类的概念.所以,要想实现 ...
- js中的操作符
写在前面 js语法 DOM对象(把body,div,p等节点树看成一个对象) BOM对象(把浏览器的地址栏历史记录DOM等装在一个对象) 浏览器是宿主,但js的宿主不限于浏览器,也可以是服务器,如no ...
- c++的单例模式及c++11对单例模式的优化
单例模式 单例模式,可以说设计模式中最常应用的一种模式了,据说也是面试官最喜欢的题目.但是如果没有学过设计模式的人,可能不会想到要去应用单例模式,面对单例模式适用的情况,可能会优先考虑使用全局或者静态 ...