本文版权归ljh2000和博客园共有,欢迎转载,但须保留此声明,并给出原文链接,谢谢合作。

本文作者:ljh2000
作者博客:http://www.cnblogs.com/ljh2000-jump/
转载请注明出处,侵权必究,保留最终解释权!

Description

为了写论文,Alex经常要整理大量的数据。 这一次,Alex面临一个严峻的考验:他需要实现一个数据结构来维护一个点集。 
现在,二维平面上有N个点。Alex 需要实现以下三种操作: 
1. 在点集里添加一个点; 
2. 给出一个点,查询它到点集里所有点的曼哈顿距离的最小值; 
3. 给出一个点,查询它到点集里所有点的曼哈顿距离的最大值。 
两个点的曼哈顿距离定义为它们的横坐标差的绝对值与纵坐标差的绝对值的和。这么困难的问题,Alex当然不会做,只好再次请你帮忙了。 

Input

第一行包含一个整数N,表示点集最初的点数。 
接下来N行,每行两个整数,依次表示每个点的横坐标和纵坐标。 
第N+2行包含一个整数Q,表示询问的数目。 
接下来Q行,每行三个整数,依次表示询问的类型,点的横坐标和纵坐标。0类型表示添加一个点,1类型表示查询到该点的曼哈顿距离的最小值,2类型表示查询最大值。 
1 ≤ N, Q ≤ 100,000,点的坐标是不超过10^9的非负整数

Output

输出若干行,依次表示每个查询操作的答案。 

Sample Input

3
7 5
6 2
3 1
5
1 6 1
1 5 5
2 7 1
0 3 2
1 1 0

Sample Output

1
2
4
3
 
 
正解:kd-tree
解题报告:
  kd-tree模板题。学习kd-tree出门右转:http://www.cnblogs.com/ljh2000-jump/tag/kd-tree/
  然而这道题我的常数无比巨大,实在找不到问题......  
 
//It is made by ljh2000
#include <iostream>
#include <cstdlib>
#include <cstring>
#include <cstdio>
#include <cmath>
#include <algorithm>
using namespace std;
typedef long long LL;
const int MAXN = 200011;
const int inf = 2000000001;
int n,m,nowD,root,ql,qr,ans;
struct KDTree{ int d[2],Min[2],Max[2],l,r; }t[MAXN];
inline bool cmp(KDTree q,KDTree qq){ if(q.d[nowD]==qq.d[nowD]) return q.d[nowD^1]<qq.d[nowD^1]; return q.d[nowD]<qq.d[nowD]; }
inline void init(int x){ for(int i=0;i<2;i++) t[x].Min[i]=t[x].Max[i]=t[x].d[i]; }
inline int count_min(int x){
int dd=0;
if(t[x].Min[0]>ql) dd+=t[x].Min[0]-ql;
if(t[x].Max[0]<ql) dd+=ql-t[x].Max[0];
if(t[x].Min[1]>qr) dd+=t[x].Min[1]-qr;
if(t[x].Max[1]<qr) dd+=qr-t[x].Max[1];
return dd;
}
inline int count_max(int x){ return max(abs(t[x].Min[0]-ql),abs(t[x].Max[0]-ql))+max(abs(t[x].Min[1]-qr),abs(t[x].Max[1]-qr)); }
inline int getint(){
int w=0,q=0; char c=getchar(); while((c<'0'||c>'9') && c!='-') c=getchar();
if(c=='-') q=1,c=getchar(); while (c>='0'&&c<='9') w=w*10+c-'0',c=getchar(); return q?-w:w;
} inline void update(int x){
int l=t[x].l,r=t[x].r;
if(l) {
if(t[l].Min[0]<t[x].Min[0]) t[x].Min[0]=t[l].Min[0];
if(t[l].Max[0]>t[x].Max[0]) t[x].Max[0]=t[l].Max[0];
if(t[l].Min[1]<t[x].Min[1]) t[x].Min[1]=t[l].Min[1];
if(t[l].Max[1]>t[x].Max[1]) t[x].Max[1]=t[l].Max[1];
}
if(r) {
if(t[r].Min[0]<t[x].Min[0]) t[x].Min[0]=t[r].Min[0];
if(t[r].Max[0]>t[x].Max[0]) t[x].Max[0]=t[r].Max[0];
if(t[r].Min[1]<t[x].Min[1]) t[x].Min[1]=t[r].Min[1];
if(t[r].Max[1]>t[x].Max[1]) t[x].Max[1]=t[r].Max[1];
}
} inline int build(int l,int r,int D){
nowD=D; int mid=(l+r)>>1; nth_element(t+l+1,t+mid+1,t+r+1,cmp);
if(l<mid) t[mid].l=build(l,mid-1,D^1);
if(mid<r) t[mid].r=build(mid+1,r,D^1);
init(mid); update(mid);
return mid;
} inline void insert(){
int u=root; int D=0;
while(1) {
if(t[u].Min[0]>t[n].d[0]) t[u].Min[0]=t[n].d[0];
if(t[u].Max[0]<t[n].d[0]) t[u].Max[0]=t[n].d[0];
if(t[u].Min[1]>t[n].d[1]) t[u].Min[1]=t[n].d[1];
if(t[u].Max[1]<t[n].d[1]) t[u].Max[1]=t[n].d[1]; if(t[n].d[D]<=t[u].d[D]) {
if(!t[u].l) { t[u].l=n; return ; }
else u=t[u].l;
}
else{
if(!t[u].r) { t[u].r=n; return ; }
else u=t[u].r;
}
D^=1;
}
} inline void query_max(int u){
int dl=0,dr=0,dd=abs(t[u].d[0]-ql)+abs(t[u].d[1]-qr); if(dd>ans) ans=dd;
if(t[u].l) dl=count_max(t[u].l); if(t[u].r) dr=count_max(t[u].r);
if(dl>dr) {
if(dl>ans) query_max(t[u].l);
if(dr>ans) query_max(t[u].r);
}
else{
if(dr>ans) query_max(t[u].r);
if(dl>ans) query_max(t[u].l);
}
} inline void query_min(int u){
int dl=inf,dr=inf,dd=abs(t[u].d[0]-ql)+abs(t[u].d[1]-qr); if(dd<ans) ans=dd;
if(t[u].l) dl=count_min(u); if(t[u].r) dr=count_min(u);
if(dl<dr) {
if(dl<ans) query_min(t[u].l);
if(dr<ans) query_min(t[u].r);
}
else{
if(dr<ans) query_min(t[u].r);
if(dl<ans) query_min(t[u].l);
}
} inline void work(){
n=getint(); for(int i=1;i<=n;i++) t[i].d[0]=getint(),t[i].d[1]=getint();
root=build(1,n,0); m=getint(); int ljh,x,y;
while(m--) {
ljh=getint(); x=getint(); y=getint();
if(ljh==0) { t[++n].d[0]=x; t[n].d[1]=y; init(n); insert(); }
else if(ljh==1) {
ql=x; qr=y; ans=abs(t[1].d[0]-ql)+abs(t[1].d[1]-qr);
query_min(root);
printf("%d\n",ans);
}
else{
ql=x; qr=y; ans=abs(t[1].d[0]-ql)+abs(t[1].d[1]-qr);
query_max(root);
printf("%d\n",ans);
}
}
} int main()
{
work();
return 0;
}

  

BZOJ4533 [BeiJing2014 WinterCamp] 数据的更多相关文章

  1. BZOJ4532: [BeiJing2014 WinterCamp] 珠链

    Description Alex喜欢玩网络游戏,认为这是智力和体力的综合锻炼.在一次游戏活动中,他意外获得了一个传说中威力极其强大的法宝:珠链.  珠链,顾名思义,就是由许多小珠子串起来的一条链.珠子 ...

  2. bzoj AC倒序

    Search GO 说明:输入题号直接进入相应题目,如需搜索含数字的题目,请在关键词前加单引号 Problem ID Title Source AC Submit Y 1000 A+B Problem ...

  3. bzoj1874 [BeiJing2009 WinterCamp]取石子游戏

    1874: [BeiJing2009 WinterCamp]取石子游戏 Time Limit: 5 Sec  Memory Limit: 162 MBSubmit: 925  Solved: 381[ ...

  4. Hadoop 中利用 mapreduce 读写 mysql 数据

    Hadoop 中利用 mapreduce 读写 mysql 数据   有时候我们在项目中会遇到输入结果集很大,但是输出结果很小,比如一些 pv.uv 数据,然后为了实时查询的需求,或者一些 OLAP ...

  5. App开发:模拟服务器数据接口 - MockApi

    为了方便app开发过程中,不受服务器接口的限制,便于客户端功能的快速测试,可以在客户端实现一个模拟服务器数据接口的MockApi模块.本篇文章就尝试为使用gradle的android项目设计实现Moc ...

  6. 使用TSQL查询和更新 JSON 数据

    JSON是一个非常流行的,用于数据交换的文本数据(textual data)格式,主要用于Web和移动应用程序中.JSON 使用“键/值对”(Key:Value pair)存储数据,能够表示嵌套键值对 ...

  7. SQL Server 大数据搬迁之文件组备份还原实战

    一.本文所涉及的内容(Contents) 本文所涉及的内容(Contents) 背景(Contexts) 解决方案(Solution) 搬迁步骤(Procedure) 搬迁脚本(SQL Codes) ...

  8. SQLSERVER将一个文件组的数据移动到另一个文件组

    SQLSERVER将一个文件组的数据移动到另一个文件组 有经验的大侠可以直接忽视这篇文章~ 这个问题有经验的人都知道怎麽做,因为我们公司的数据量不大没有这个需求,也不知道怎麽做实验 今天求助了QQ群里 ...

  9. 【.net 深呼吸】设置序列化中的最大数据量

    欢迎收看本期的<老周吹牛>节目,由于剧组严重缺钱,故本节目无视频无声音.好,先看下面一个类声明. [DataContract] public class DemoObject { [Dat ...

随机推荐

  1. Echarts xAxis boundaryGap

    Echarts  xAxis----->boundaryGap: false 坐标轴两边留白策略,类目轴和非类目轴的设置和表现不一样. 类目轴中 boundaryGap 可以配置为 true 和 ...

  2. 使用 Fiddler 上传微信公众账号 自定义菜单

    0.你必须有微信公众账号的服务号.成为开发者之后.... 1.得到你的 appid (xxxxxxoooo)和 secret (oooooooxxxxxxx) 2.用这个链接得到你的 access_t ...

  3. Oracle 释放flash recovery area的四种方法

    早上收到一台Linux服务器磁盘告警邮件以及监控告警日志程序发来的邮件.检查过后,发现Linux服务器中一个分区没有空间了.主要原因是由于昨晚程序员做升级时,产生了大量的归档日志,导致联机重做日志无法 ...

  4. Java 6 JVM参数选项大全(中文版)

    原文来自: http://kenwublog.com/docs/java6-jvm-options-chinese-edition.htm 本文是基于最新的SUN官方文档Java SE 6 Hotsp ...

  5. audacity开源VS2013环境搭建

    audacity是非常不错的音频开源,其中音频效果处理的种类很多,非常方便借鉴和研究. 但是audacity的界面库是使用wxWidgets(一个跨平台的界面库),配置过程中需要折腾一下. 1,首先去 ...

  6. monkeyrunner之录制与回放(七)

    monkeyrunner为我们提供了录制 回放的功能. 录制与回放使用原因:实际项目,需求变更频繁,且测试任务多,我们没有足够时间去写测试脚本,这是就可以进行录制脚本,然后通过回放,跑完需要的流程. ...

  7. 解决cefsharp在winform中不显示tooltipText问题(网页元素的title提示)

    1.监听网页属性改变事件 webView.PropertyChanged += webView_PropertyChanged; 2.拖一个ToolTip控件到窗体 3.在webView_Proper ...

  8. 一种可实时处理 O(1)复杂度图像去雾算法的实现。

    在我博文的一系列的文章,有不少算法都于去雾有关,比如限制对比度自适应直方图均衡化算法原理.实现及效果.局部自适应自动色阶/对比度算法在图像增强上的应用这两个增强算法都有一定的去雾能力,而最直接的就是& ...

  9. POJ 1273 Drainage Ditches题解——S.B.S.

    Drainage Ditches Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 67823   Accepted: 2620 ...

  10. Spark的精简安装步骤---陈楠心血总结

    1.下载解压 (1)安装Scala-2.10.4并解压 (2)/etc/profile加入PATH路径 (3)source /etc/profile使PATH的配置生效 (4)下载并解压spark-1 ...