【BZOJ4066】简单题(KD-Tree)
【BZOJ4066】简单题(KD-Tree)
题面
题解
如果这题不卡空间,并且不强制在线的话
显然可以用\(CDQ\)分治做
但是它又卡空间又强制在线,于是我们欢快的来用\(KD-Tree\)吧。
用\(KD-Tree\)维护每一个点,每次询问的时候
判断询问的矩形和当前矩形的交
如果全部覆盖直接加上所有的和
如果没有交则直接返回
如果有部分交则递归处理。
我用了两种方案防止\(KD-Tree\)被卡
第一种:定时重构。大概就是每插入\(10000\)次后就重构一次
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<set>
#include<map>
#include<vector>
#include<queue>
using namespace std;
#define ll long long
#define RG register
#define MAX 100200
#define mid o
#define ls (t[o].ch[0])
#define rs (t[o].ch[1])
#define cmin(a,b) (a<b?a:a=b)
#define cmax(a,b) (a>b?a:a=b)
inline int read()
{
RG int x=0,t=1;RG char ch=getchar();
while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
if(ch=='-')t=-1,ch=getchar();
while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
return x*t;
}
const double alpha=0.762333;
int D;
struct Node{int d[2],v;}a[MAX];
bool operator<(Node a,Node b){return a.d[D]<b.d[D];}
struct KD_Node
{
int mn[2],mx[2],ch[2];
Node a;
int sum;
void clear(){mn[0]=mn[1]=mx[0]=mx[1]=a.d[0]=a.d[1]=a.v=sum=0;}
}t[MAX];
int tot,Num;
void update(int x,int y)
{
cmin(t[x].mn[0],t[y].mn[0]);cmin(t[x].mn[1],t[y].mn[1]);
cmax(t[x].mx[0],t[y].mx[0]);cmax(t[x].mx[1],t[y].mx[1]);
}
void pushup(int o)
{
t[o].mn[0]=t[o].mx[0]=t[o].a.d[0];
t[o].mn[1]=t[o].mx[1]=t[o].a.d[1];
t[o].sum=t[ls].sum+t[rs].sum+t[o].a.v;
if(ls)update(o,ls);if(rs)update(o,rs);
}
int Build(int l,int r,int nd)
{
int mid=(l+r)>>1;D=nd;
nth_element(&a[l],&a[mid],&a[r+1]);
t[o].a=a[mid];
if(l<mid)ls=Build(l,mid-1,nd^1);else ls=0;
if(r>mid)rs=Build(mid+1,r,nd^1);else rs=0;
pushup(o);return o;
}
void insert(int &o,int nd,Node a)
{
if(!o){o=tot;t[o].a=a;pushup(o);return;}
if(t[o].a.d[nd]<a.d[nd])insert(rs,nd^1,a);
else insert(ls,nd^1,a);
pushup(o);
}
int checksame(KD_Node a,int x1,int x2,int y1,int y2)
{
if(x1<=a.mn[0]&&a.mx[0]<=x2&&y1<=a.mn[1]&&a.mx[1]<=y2)return 1;//all
if(x2<a.mn[0]||a.mx[0]<x1)return 0;
if(y2<a.mn[1]||a.mx[1]<y1)return 0;
return 2;
}
int Query(int o,int x1,int x2,int y1,int y2)
{
int ret=0;
if(t[o].a.d[0]>=x1&&t[o].a.d[0]<=x2&&t[o].a.d[1]>=y1&&t[o].a.d[1]<=y2)
ret+=t[o].a.v;
if(ls)
{
int d=checksame(t[ls],x1,x2,y1,y2);
if(d==1)ret+=t[ls].sum;
if(d==2)ret+=Query(ls,x1,x2,y1,y2);
}
if(rs)
{
int d=checksame(t[rs],x1,x2,y1,y2);
if(d==1)ret+=t[rs].sum;
if(d==2)ret+=Query(rs,x1,x2,y1,y2);
}
return ret;
}
int N,rt;
int main()
{
N=read();
int lans=0,opt,x,y,A,x1,y1,x2,y2;
while(233)
{
opt=read();
if(opt==3)break;
if(opt==1)
{
x=read()^lans,y=read()^lans,A=read()^lans;
insert(rt,0,a[++tot]=(Node){x,y,A});
if(tot%10000==0)rt=Build(1,tot,0);
}
else
{
x1=read()^lans,y1=read()^lans,x2=read()^lans,y2=read()^lans;
printf("%d\n",lans=Query(rt,x1,x2,y1,y2));
}
}
}
第二种:类似替罪羊树,当不满足平衡因子时,把子树拍掉重构。
#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<cmath>
#include<algorithm>
#include<set>
#include<map>
#include<vector>
#include<queue>
using namespace std;
#define ll long long
#define RG register
#define MAX 200200
#define ls (t[o].ch[0])
#define rs (t[o].ch[1])
#define cmin(a,b) (a<b?a:a=b)
#define cmax(a,b) (a>b?a:a=b)
inline int read()
{
RG int x=0,t=1;RG char ch=getchar();
while((ch<'0'||ch>'9')&&ch!='-')ch=getchar();
if(ch=='-')t=-1,ch=getchar();
while(ch<='9'&&ch>='0')x=x*10+ch-48,ch=getchar();
return x*t;
}
const double alpha=0.75;
int D;
struct Node{int d[2],v;}a[MAX];
bool operator<(Node a,Node b){return a.d[D]<b.d[D];}
struct KD_Node
{
int mn[2],mx[2],ch[2];
Node a;
int sum,size;
void clear(){mn[0]=mn[1]=mx[0]=mx[1]=a.d[0]=a.d[1]=a.v=sum=size=0;}
}t[MAX];
int Q[MAX],top;
int tot,Num;
int NewNode()
{
if(!top)return ++tot;
t[Q[top]].clear();
return Q[top--];
}
void update(int x,int y)
{
cmin(t[x].mn[0],t[y].mn[0]);cmin(t[x].mn[1],t[y].mn[1]);
cmax(t[x].mx[0],t[y].mx[0]);cmax(t[x].mx[1],t[y].mx[1]);
}
void pushup(int o)
{
t[o].mn[0]=t[o].mx[0]=t[o].a.d[0];
t[o].mn[1]=t[o].mx[1]=t[o].a.d[1];
t[o].size=t[ls].size+t[rs].size+1;
t[o].sum=t[ls].sum+t[rs].sum+t[o].a.v;
if(ls)update(o,ls);if(rs)update(o,rs);
}
int Build(int l,int r,int nd)
{
int mid=(l+r)>>1;D=nd;
nth_element(&a[l],&a[mid],&a[r+1]);
int o=NewNode();
t[o].a=a[mid];
if(l<mid)ls=Build(l,mid-1,nd^1);else ls=0;
if(r>mid)rs=Build(mid+1,r,nd^1);else rs=0;
pushup(o);return o;
}
void kill(int o){if(ls)kill(ls);a[++Num]=t[o].a;Q[++top]=o;if(rs)kill(rs);}
void checker(int &o,int nd)
{
if(alpha*t[o].size<t[ls].size||alpha*t[o].size<t[rs].size)
Num=0,kill(o),o=Build(1,Num,nd);
}
void insert(int &o,int nd,Node a)
{
if(!o){o=NewNode();t[o].a=a;pushup(o);return;}
if(t[o].a.d[nd]<a.d[nd])insert(rs,nd^1,a);
else insert(ls,nd^1,a);
pushup(o);checker(o,nd);
}
int checksame(KD_Node a,int x1,int x2,int y1,int y2)
{
if(x1<=a.mn[0]&&a.mx[0]<=x2&&y1<=a.mn[1]&&a.mx[1]<=y2)return 1;//all
if(x2<a.mn[0]||a.mx[0]<x1)return 0;
if(y2<a.mn[1]||a.mx[1]<y1)return 0;
return 2;
}
int Query(int o,int x1,int x2,int y1,int y2)
{
int ret=0;
if(t[o].a.d[0]>=x1&&t[o].a.d[0]<=x2&&t[o].a.d[1]>=y1&&t[o].a.d[1]<=y2)
ret+=t[o].a.v;
if(ls)
{
int d=checksame(t[ls],x1,x2,y1,y2);
if(d==1)ret+=t[ls].sum;
if(d==2)ret+=Query(ls,x1,x2,y1,y2);
}
if(rs)
{
int d=checksame(t[rs],x1,x2,y1,y2);
if(d==1)ret+=t[rs].sum;
if(d==2)ret+=Query(rs,x1,x2,y1,y2);
}
return ret;
}
int N,rt;
int main()
{
N=read();
int lans=0,opt,x,y,A,x1,y1,x2,y2;
while(233)
{
opt=read();
if(opt==3)break;
if(opt==1)
{
x=read()^lans,y=read()^lans,A=read()^lans;
insert(rt,0,(Node){x,y,A});
}
else
{
x1=read()^lans,y1=read()^lans,x2=read()^lans,y2=read()^lans;
printf("%d\n",lans=Query(rt,x1,x2,y1,y2));
}
}
}
【BZOJ4066】简单题(KD-Tree)的更多相关文章
- BZOJ4066:简单题(K-D Tree)
Description 你有一个N*N的棋盘,每个格子内有一个整数,初始时的时候全部为0,现在需要维护两种操作: 命令 参数限制 内容 1 x y A 1<=x,y<=N,A是正整数 ...
- P4148 简单题 k-d tree
思路:\(k-d\ tree\) 提交:2次 错因:整棵树重构时的严重错误:没有维护父子关系(之前写的是假重构所以没有维护父子关系) 题解: 遇到一个新的点就插进去,如果之前出现过就把权值加上. 代码 ...
- [BZOJ2683][BZOJ4066]简单题
[BZOJ2683][BZOJ4066]简单题 试题描述 你有一个N*N的棋盘,每个格子内有一个整数,初始时的时候全部为0,现在需要维护两种操作: 命令 参数限制 内容 1 x y A 1<=x ...
- bzoj4066: 简单题 K-Dtree
bzoj4066: 简单题 链接 bzoj 思路 强制在线.k-dtree. 卡常啊.空间开1e6就T了. 代码 #include <bits/stdc++.h> #define my_m ...
- Bzoj4066 简单题
Time Limit: 50 Sec Memory Limit: 20 MBSubmit: 2185 Solved: 581 Description 你有一个N*N的棋盘,每个格子内有一个整数,初 ...
- bzoj 4066: 简单题 K-D树
题目大意: http://www.lydsy.com/JudgeOnline/problem.php?id=4066 题解 我们把每次的修改操作都当作二维平面上多了一个权值点 对于每组询问可以看做求一 ...
- BZOJ4066 简单题(KD-Tree)
板子题. #include<iostream> #include<cstdio> #include<cmath> #include<cstdlib> # ...
- 【kd-tree】bzoj4066 简单题
同p1176. #include<cstdio> #include<cmath> #include<algorithm> using namespace std; ...
- bzoj 4066 & bzoj 2683 简单题 —— K-D树(含重构)
题目:https://www.lydsy.com/JudgeOnline/problem.php?id=4066 https://www.lydsy.com/JudgeOnline/problem.p ...
- 初涉k-d tree
听说k-d tree是一个骗分的好东西?(但是复杂度差评??? 还听说绍一的kdt常数特别小? KDT是什么 KDT的全称是k-degree tree,顾名思义,这是一种处理多维空间的数据结构. 例如 ...
随机推荐
- HBase 数据的多版本特性潜在的意外
HBase做为KeyValue结构存储,在存储上是依照RowKey的字典序进行排序,对于很多应用而言这可能远远不够,好在HBase的数据可以存储多个版本,并且版本可以排序,其理论上最大的版本数目Int ...
- PS 证件照换颜色
1.打开要修改的图片,然后先Ctrl+J备份一份 2.点击魔法棒,点击要换颜色的地方,如衣服,之后会出现虚线,如果自动选择的不全,可以按住Shift键自行选择区域 3.然后Shift+Fn+F5(由于 ...
- Appium(Python)测试混血App
Hybrid App(混合模式移动应用)是指介于web-app.native-app这两者之间的app兼具Native App良好用户交互体验的优势和Web App跨平台开发的优势 HybridApp ...
- Web自动化selenium技术快速实现爬虫
selenium是大家众所周知的web自动化测试框架,主要用来完成web网站项目的自动化测试,但其实如果要实现一个web爬虫,去某些网站爬取数据,其实用selenium来实现也很方便. 比如,我们现在 ...
- 6.2 element和elements
为什么这个要单独拿出来说,因为我在很多群里面看见很多人不能区分这个! 因为之前的包有点问题,另外后续还会更换app,因为部分app可能没有符合的案例场景,我需要找到那个场景给大家做个实例..便于大家跟 ...
- 第5章 Linux网络编程基础
第5章 Linux网络编程基础 5.1 socket地址与API 一.理解字节序 主机字节序一般为小端字节序.网络字节序一般为大端字节序.当格式化的数据在两台使用了不同字节序的主机之间直接传递时,接收 ...
- 【WXS全局对象】Number
属性: 名称 说明 Number.MAX_VALUE 返回JS中可表示的最大的数.它的近似值为 1.7976931348623157 x 10308. Number.MIN_VALUE 返回JS中可表 ...
- Android开发-API指南-<permission-tree>
<permission-tree> 英文原文:http://developer.android.com/guide/topics/manifest/permission-tree-elem ...
- (转)Android SearchView 搜索框
如果对这个效果感觉不错, 请往下看. 背景: 天气预报app, 本地数据库存储70个大中城市的基本信息, 根据用户输入的或通过搜索框选取的城市, 点击查询按钮后, 异步请求国家气象局数据, 得到返回的 ...
- android入门 — PopupWindow
第一步:自定义xml文件 第二步:获取LayoutInflater对象 第三步:调用inflate()方法获取View对象 第四步:创建PopupWindow对象 第五步:调用PopupWindow的 ...