题意: 给一个矩阵,每次查询一个子矩阵内的最大最小值,然后更新子矩阵中心点为(Max+Min)/2.

解法: 由于是矩阵,且要求区间最大最小和更新单点,很容易想到二维的线段树,可是因为之前没写过二维的线段树,所以没跳出来。后来熟悉了一下,原来很多细节地方都没有考虑到。

这里build,update,query都分为两个函数,第一个为Y轴的(sub_update),第二个为X轴的(update),不仅每个sub_update或sub_build后面要加Y轴的pushup函数,而且每个update或build函数最后也要进行pushup,只不过是X轴的pushup,写成 sub_build(rt,-1,1,n,1) 或 sub_update(rt,-1,1,n,1);

二维线段树就是这里比较费解一点,我感觉, 其他地方还比较好理解。

代码:

#include <iostream>
#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <cmath>
#include <algorithm>
#define Mod 1000000007
#define lll __int64
using namespace std;
#define N 100007 struct sub_seg{
lll maxi,mini;
}; struct seg{
sub_seg st[<<];
}tt[<<];
int n;
lll Max,Min; void pushup(int xrt,int rt)
{
tt[xrt].st[rt].maxi = max(tt[xrt].st[rt<<].maxi,tt[xrt].st[rt<<|].maxi);
tt[xrt].st[rt].mini = min(tt[xrt].st[rt<<].mini,tt[xrt].st[rt<<|].mini);
} void sub_build(int xrt,int posY,int l,int r,int rt)
{
if(l == r)
{
if(posY != -)
{
scanf("%I64d",&tt[xrt].st[rt].maxi);
tt[xrt].st[rt].mini = tt[xrt].st[rt].maxi;
}
else
{
tt[xrt].st[rt].maxi = max(tt[xrt<<].st[rt].maxi,tt[xrt<<|].st[rt].maxi);
tt[xrt].st[rt].mini = min(tt[xrt<<].st[rt].mini,tt[xrt<<|].st[rt].mini);
}
return;
}
int mid = (l + r) >> ;
sub_build(xrt,posY, l,mid, *rt);
sub_build(xrt,posY, mid+,r, *rt+);
pushup(xrt,rt); //pushup Y
} void build(int l, int r, int rt)
{
if(l == r)
{
sub_build(rt,l,,n,);
return;
}
int mid = (l + r) >> ;
build(l,mid,*rt);
build(mid+,r,*rt+);
sub_build(rt,-,,n,); //pushup X
} void sub_update(int xrt,int pos,int y,int l,int r,int val,int rt)
{
if(l == r)
{
if(pos!=-)
tt[xrt].st[rt].maxi = tt[xrt].st[rt].mini = val;
else
{
tt[xrt].st[rt].maxi = max(tt[xrt<<].st[rt].maxi,tt[xrt<<|].st[rt].maxi);
tt[xrt].st[rt].mini = min(tt[xrt<<].st[rt].mini,tt[xrt<<|].st[rt].mini);
}
return;
}
int mid = (l+r) >> ;
if(y <= mid) sub_update(xrt, pos, y, l,mid, val, rt<<) ;
else sub_update(xrt, pos, y, mid+,r, val, rt<<|) ;
pushup(xrt,rt); //pushup Y维
} void update(int x,int y,int l,int r,int val,int rt) //单点更新
{
if(l == r)
{
sub_update(rt,l,y,,n,val,);
return;
}
int mid = (l+r) >> ;
if(x<=mid) update(x,y, l,mid, val, rt<<);
else update(x,y, mid+,r, val, rt<<|);
sub_update(rt,-,y,,n,val,); //pushup X维
} void sub_query(int xrt,int aa,int bb,int l,int r,int rt)
{
if(aa <= l && bb >= r)
{
Max = max(Max,tt[xrt].st[rt].maxi);
Min = min(Min,tt[xrt].st[rt].mini);
return;
}
int mid = (l+r)>>;
if(aa<=mid) sub_query(xrt,aa,bb,l,mid, rt<<);
if(bb>mid) sub_query(xrt,aa,bb,mid+,r, rt<<|);
} void query(int l,int r,int aa,int bb,int L1,int R1,int rt)
{
if(aa <= l && bb >= r)
{
sub_query(rt,L1,R1,,n,);
return;
}
int mid = (l+r)>>;
if(aa <= mid) query(l,mid,aa,bb,L1,R1, rt<<);
if(bb > mid) query(mid+,r,aa,bb,L1,R1, rt<<|);
} int main()
{
int t,i,j,m,x,y,L,cs = ;
scanf("%d",&t);
while(t--)
{
scanf("%d",&n);
build(,n,);
scanf("%d",&m);
printf("Case #%d:\n",cs++);
while(m--)
{
scanf("%d%d%d",&x,&y,&L);
int Lx = max(,x-L/);
int Ly = max(,y-L/);
int Rx = min(n,x+L/);
int Ry = min(y+L/,n);
Max = -, Min = Mod;
query(,n,Lx,Rx,Ly,Ry,);
lll mid = (Max+Min)/2LL;
printf("%I64d\n",mid);
update(x,y,,n,mid,);
}
}
return ;
}

HDU 4819 Mosaic --二维线段树(树套树)的更多相关文章

  1. HDU 4819 Mosaic 二维线段树

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

  2. UVALive 6709 - Mosaic 二维线段树

    题目链接 给一个n*n的方格, 每个方格有值. 每次询问, 给出三个数x, y, l, 求出以x, y为中心的边长为l的正方形内的最大值与最小值, 输出(maxx+minn)/2, 并将x, y这个格 ...

  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 (二维线段树&区间最值)题解

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

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

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

  7. hdu 4819 二维线段树模板

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

  8. HDU 4819 二维线段树

    13年长春现场赛的G题,赤裸裸的二维线段树,单点更新,区间查询 不过我是第一次写二维的,一开始写T了,原因是我没有好好利用行段,说白一点,还是相当于枚举行,然后对列进行线段树,那要你写二维线段树干嘛 ...

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

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

随机推荐

  1. [moka同学笔记]yii2 activeForm 表单样式的修改(二)

    aaarticlea/png;base64,iVBORw0KGgoAAAANSUhEUgAABAEAAANXCAIAAADLkdErAAAgAElEQVR4nOzdfWwc953nef6zwO5Zg8

  2. double 类型转化为Integer类型 ---DecimalFormat

    假设x是你要转换的double类型变量: 不进行四舍五入操作: (int)x 进行四舍五入操作: Integer.parseInt(new java.text.DecimalFormat(" ...

  3. mvc与三层结构终极区别

    http://blog.csdn.net/csh624366188/article/details/7183872 http://www.cnblogs.com/zhhh/archive/2011/0 ...

  4. [Azure] Notification Hubs注册模式

    [Azure] Notification Hubs注册模式 关于Azure Notification Hubs的注册模式,可以参考下列连结的文件内容. Notification Hubs Featur ...

  5. bash shell命令(2)

    在上篇<bash shell命令(1)>中,介绍了几种简单的linux shell命令,今天继续介绍bash shell命令 本文地址:http://www.cnblogs.com/arc ...

  6. iOS设计模式之工厂方法模式

    工厂方法模式 基本理解 工厂方法模式:定义一个用于创建对象的接口,让子类决定实例化哪一个类.工厂方法使一个类的实例化延迟到其子类. 简单工厂的最大优点就是工厂类中包含了必要的逻辑判断,根据客户端的选择 ...

  7. 我对XCode Objective-c Cocoa的简单理解

    我对XCode Objective-c Cocoa的简单理解 Xcode Xcode说的通俗一点就是开发OS X 和 iOS 应用程序的. 如果我们想要认真点说 ,Xcode 是运行在操作系统Mac ...

  8. .NET下的并行开发

    并行开发一直是程序员在开发项目中遇到的一道坎,但为了迎合硬件的升级,面对高端多核的处理器,并行编程势在必行.在.NET平台下的开发支持并行模式,下面用一个实际项目说明并行的高效率和神奇之处. 在优化中 ...

  9. C语言-03-流程控制

    一.选择结构 1> if语句 使用注意 ① if语句中的条件语句,不要把==和=弄混,可以把常量作为左值, 这样的话,在无用=的情况下,编译时会报错 ② if语句后若要定义新的变量或者有多条语句 ...

  10. javascript 构造函数方式定义对象

    javascript是动态语言,可以在运行时给对象添加属性,也可以给对象删除(delete)属性 <html> <head> <script type="tex ...