13年长春现场赛的G题,赤裸裸的二维线段树,单点更新,区间查询

不过我是第一次写二维的,一开始写T了,原因是我没有好好利用行段,说白一点,还是相当于枚举行,然后对列进行线段树,那要你写二维线段树干嘛

二维就是在每个行段也建一棵树,来代表这个区间的行里的某些列的值

其他操作倒是不难,因为有一维的功底,只是多写一个,刷刷刷就出来了

就是更新操作的时候有点麻烦,up函数不好写,必须先更新底层,复层是区间值,无法先进行更新,然后底层向父层转移也是有点小技巧,因为每个行段点里面的某些列的列端点号肯定是相同的,比如 我dp[1][rt]和dp[2][rt],表示的都是同样的列,只是一个是儿子行,一个是父亲行,父亲行是>儿子行的,所以更新到底层的时候,就往上每次对第一维除2来更新父节点的相关区域即可

还有发现这个不好直接复制,本题是维护最小值和最大值,在建树的时候,直接修改不太好,先把所有的点的最大设置为-inf,最小设置为INF,这样以修改的方式去赋初值,包括之后的修改也是这样。

#include <iostream>
#include <cstdio>
#include <cstring>
#include <algorithm>
#define lson rt<<1,l,mid
#define rson rt<<1|1,mid+1,r
#define INF 1<<30
using namespace std;
const int N = 810;
int ds[N*3][N*3];
int db[N*3][N*3];
int flag[N*3][N*3];
int n;
int A[N][N];
struct node
{
int mini,maxn;
};
void up(int k,int rt)
{
db[k][rt]=max(db[k][rt<<1],db[k][rt<<1|1]);
ds[k][rt]=min(ds[k][rt<<1],ds[k][rt<<1|1]);
for (int i=(k>>1);i;i>>=1){
db[i][rt]=max(db[i<<1][rt],db[i<<1|1][rt]);
ds[i][rt]=min(ds[i<<1][rt],ds[i<<1|1][rt]);
}
}
void buildc(int k,int rt,int l,int r)
{
db[k][rt]=-INF;
ds[k][rt]=INF;
if (l>=r){
return;
}
int mid=(l+r)>>1;
buildc(k,lson);
buildc(k,rson);
}
void buildr(int rt,int l,int r)
{
buildc(rt,1,1,n);
if (l>=r){
return;
}
int mid=(l+r)>>1;
buildr(lson);
buildr(rson);
}
node queryc(int k,int c1,int c2,int rt,int l,int r)
{
if (c1<=l && r<=c2){
node x;
x.maxn=db[k][rt];
x.mini=ds[k][rt];
return x;
}
int mid=(l+r)>>1;
if (mid<c1){
return queryc(k,c1,c2,rson);
}
else
if (mid>=c2){
return queryc(k,c1,c2,lson);
}
else{
node a=queryc(k,c1,c2,lson);
node b=queryc(k,c1,c2,rson);
node c;
c.mini=min(a.mini,b.mini);
c.maxn=max(a.maxn,b.maxn);
return c;
}
}
node queryr(int r1,int r2,int c1,int c2,int rt,int l,int r)
{
if (r1<=l && r<=r2)
{
return queryc(rt,c1,c2,1,1,n);
}
int mid=(l+r)>>1;
if (r2<=mid){
return queryr(r1,r2,c1,c2,lson);
}
else
if (r1>mid){
return queryr(r1,r2,c1,c2,rson);
}
else{
node a=queryr(r1,r2,c1,c2,lson);
node b=queryr(r1,r2,c1,c2,rson);
node c;
c.maxn=max(a.maxn,b.maxn);
c.mini=min(a.mini,b.mini);
return c;
}
} void fixc(int val,int k,int C,int rt,int l,int r)
{
if (l>=r)
{
db[k][rt]=val;
ds[k][rt]=val;
for (int i=(k>>1);i;i>>=1){
db[i][rt]=max(db[i<<1][rt],db[i<<1|1][rt]);
ds[i][rt]=min(ds[i<<1][rt],ds[i<<1|1][rt]);
}
return;
}
int mid=(l+r)>>1;
if (C<=mid) fixc(val,k,C,lson);
else fixc(val,k,C,rson);
up(k,rt);
}
void fixr(int val,int R,int C,int rt,int l,int r)
{
if (l>=r)
{
fixc(val,rt,C,1,1,n);
return;
}
int mid=(l+r)>>1;
if (R<=mid) fixr(val,R,C,lson);
else fixr(val,R,C,rson);
}
int main()
{
int t,kase=0;
scanf("%d",&t);
while (t--)
{
scanf("%d",&n);
buildr(1,1,n);
for (int i=1;i<=n;i++){
for (int j=1;j<=n;j++){
scanf("%d",&A[i][j]);
fixr(A[i][j],i,j,1,1,n);
}
} int Q,L,R,S;
scanf("%d",&Q);
printf("Case #%d:\n",++kase);
int r1,r2,c1,c2;
while (Q--)
{
scanf("%d%d%d",&L,&R,&S);
r1=max(L-S/2,1);
r2=min(L+S/2,n);
c1=max(R-S/2,1);
c2=min(R+S/2,n);
node ans=queryr(r1,r2,c1,c2,1,1,n);
int ret=(ans.maxn+ans.mini)/2;
printf("%d\n",ret);
fixr(ret,L,R,1,1,n);
}
}
return 0;
}

  

HDU 4819 二维线段树的更多相关文章

  1. hdu 4819 二维线段树模板

    /* HDU 4819 Mosaic 题意:查询某个矩形内的最大最小值, 修改矩形内某点的值为该矩形(Mi+MA)/2; 二维线段树模板: 区间最值,单点更新. */ #include<bits ...

  2. Mosaic HDU 4819 二维线段树入门题

    Mosaic Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 102400/102400 K (Java/Others)Total S ...

  3. HDU 4819 Mosaic(13年长春现场 二维线段树)

    HDU 4819 Mosaic 题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4819 题意:给定一个n*n的矩阵,每次给定一个子矩阵区域(x,y,l) ...

  4. HDU 4819 Mosaic (二维线段树)

    Mosaic Time Limit: 10000/5000 MS (Java/Others)    Memory Limit: 102400/102400 K (Java/Others)Total S ...

  5. HDU 4819 Mosaic --二维线段树(树套树)

    题意: 给一个矩阵,每次查询一个子矩阵内的最大最小值,然后更新子矩阵中心点为(Max+Min)/2. 解法: 由于是矩阵,且要求区间最大最小和更新单点,很容易想到二维的线段树,可是因为之前没写过二维的 ...

  6. HDU 4819 Mosaic 二维线段树

    Mosaic Time Limit: 1 Sec Memory Limit: 256 MB 题目连接 http://acm.hust.edu.cn/vjudge/contest/view.action ...

  7. HDU 4819 Mosaic (二维线段树&区间最值)题解

    思路: 二维线段树模板题,马克一下,以后当模板用 代码: #include<cstdio> #include<cmath> #include<cstring> #i ...

  8. HDU 4819 Mosaic 【二维线段树】

    题目大意:给你一个n*n的矩阵,每次找到一个点(x,y)周围l*l的子矩阵中的最大值a和最小值b,将(x,y)更新为(a+b)/2 思路:裸的二维线段树 #include<iostream> ...

  9. HDU 1823 Luck and Love(二维线段树)

    之前只知道这个东西的大概概念,没具体去写,最近呵呵,今补上. 二维线段树 -- 点更段查 #include <cstdio> #include <cstring> #inclu ...

随机推荐

  1. ProtoBuf开发者指南

    目录 1   概览 1.1   什么是protocol buffer 1.2   他们如何工作 1.3   为什么不用XML? 1.4   听起来像是为我的解决方案,如何开始? 1.5   一点历史 ...

  2. Scrapy 下载文件和图片

    我们学习了从网页中爬取信息的方法,这只是爬虫最典型的一种应用,除此之外,下载文件也是实际应用中很常见的一种需求,例如使用爬虫爬取网站中的图片.视频.WORD文档.PDF文件.压缩包等. 1.Files ...

  3. Vue下URL地址栏参数改变却不能刷新界面

    在完成毕业设计(基于Vue的信息资讯展示与管理平台)的过程中,处理如下图所示的 点击左侧栏目列表跳转到对应文章列表 的问题时,初次点击可以跳转到对应的页面,但是当第二次点击时,虽然地址栏的参数改变了, ...

  4. natapp 内网穿透服务

    参考文章:https://www.jianshu.com/p/91a321e584b8 参考文章:https://www.jianshu.com/p/c4cb8666c96a 一.什么是内网穿透 通过 ...

  5. new Vue() 和 export default {}及Vue页面组件和标签组件说明与比较(非常重要)

    说明与比较:new Vue() 和 export default {} (1)vue就是一个构造函数 (2)vue标签组件:是HTML标签的扩展https://www.cnblogs.com/w-wa ...

  6. eclipse如何把多个项目放在文件夹下,用文件夹分开不同的项目

    在Package Explorer顶部的右侧有有机表图标按钮,点击倒三角 Top Level Elements->Working Set.此时就会发现,很多项目会自动纳入一个文件夹,这个文件夹的 ...

  7. 解题报告:luogu P1516 青蛙的约会

    题目链接:P1516 青蛙的约会 考察拓欧与推式子\(qwq\). 题意翻译? 求满足 \[\begin{cases}md+x\equiv t\pmod{l}\\nd+y\equiv t\pmod{l ...

  8. jdk环境

    安装方式一  jdk环境 干净的环境 将tar包解压到 /usr/local下 版本为jdk-8u211-linux-x64.tar.gz  路径为/usr/local/jdk1.8.0_211 /u ...

  9. 5G/NR OTA (Over The Air) 测试详解

    原文链接:http://www.sharetechnote.com/html/5G/5G_OTA.html 1 什么是OTA (Over The Air) OTA代表Over The Air.为了使用 ...

  10. node - 处理跨域 ( 两行代码解决 )

    1,安装 cors 模块 : npm install cors 2,代码 : var express = require('express') var app = express() var cors ...