luoguP3769 [CH弱省胡策R2]TATT
luoguP3769 [CH弱省胡策R2]TATT
PS:做这题前先切掉 P4148简单题,对于本人这样的juruo更助于理解,当然dalao就当练练手吧
题目大意: 现在有n个四维空间中的点,请求出一条最长的路径,满足任意一维坐标都是单调不降的
偏模板的K-D Tree
题目没规定起点,则从任意一点出发,按维度优先级以及每个维度坐标为关键字排序,
每点作为终点查询一次,再插入,其他就是模板化的代码了
这里用到了一个小技巧,就是初始化将子树0的值赋值,避免过多的特判,使代码更加简洁
for(LL i=0;i<=3;++i)
tree[0].mi[i]=inf,tree[0].mx[i]=-inf; tree[0].maxn=-inf;
My complete code:
#include<cstdio>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<iostream>
using namespace std;
typedef long long LL;
const LL maxn=1e5+10;
const LL inf=1e9;
inline LL read(){
LL x=0,f=1; char c=getchar();
while(c<'0'||c>'9'){
if(c=='-') f=-1; c=getchar();
}
while(c>='0'&&c<='9'){
x=x*10+c-'0'; c=getchar();
}
return x*f;
}
struct node{
LL son[2],d[4],maxn,size,val,mi[4],mx[4];
}tree[maxn],t2[maxn],tmp;
bool operator < (node g1,node g2){
for(LL i=0;i<=3;++i){
if(g1.d[i]<g2.d[i]) return 1;
if(g1.d[i]>g2.d[i]) return 0;
}return 0;
}
LL n,ans,ans_maxn,top,cnt,root,WD; LL sta[maxn];
inline void update(LL now){
LL son0=tree[now].son[0],son1=tree[now].son[1];
for(LL i=0;i<=3;++i){
tree[now].mi[i]=min(min(tree[son0].mi[i],tree[son1].mi[i]),tree[now].d[i]);
tree[now].mx[i]=max(max(tree[son0].mx[i],tree[son1].mx[i]),tree[now].d[i]);
}
tree[now].maxn=max(tree[now].val,max(tree[son0].maxn,tree[son1].maxn));
tree[now].size=tree[son0].size+tree[son1].size+1;
}
inline bool cmp(LL g1,LL g2){
return tree[g1].d[WD]<tree[g2].d[WD];
}
LL build(LL l,LL r,LL wd){
if(l>r)
return 0;
LL mid=(l+r)>>1;
WD=wd;
nth_element(sta+l,sta+mid,sta+r+1,cmp);
LL now=sta[mid];
tree[now].son[0]=build(l,mid-1,(wd+1)%4);
tree[now].son[1]=build(mid+1,r,(wd+1)%4);
update(now);
return now;
}
void pai(LL now){
if(!now)
return;
sta[++top]=now;
LL son0=tree[now].son[0],son1=tree[now].son[1];
pai(son0); pai(son1);
}
inline void check(LL &now,LL wd){
LL son0=tree[now].son[0],son1=tree[now].son[1];
if(tree[now].size*0.75<tree[son0].size||tree[now].size*0.75<tree[son1].size){
top=0;
pai(now);
now=build(1,tree[now].size,wd);
}
}
void insert(LL &now,LL wd){
if(!now){
now=++cnt;
tree[now]=tmp;
for(LL i=0;i<=3;++i)
tree[now].mi[i]=tree[now].mx[i]=tree[now].d[i];
tree[now].maxn=tree[now].val; tree[now].size=1;
return;
}
if(tmp.d[wd]<tree[now].d[wd])
insert(tree[now].son[0],(wd+1)%4);
else
insert(tree[now].son[1],(wd+1)%4);
update(now);
check(now,wd);
} inline LL dis(node g1,node g2){
for(LL i=0;i<=3;++i)
if(g1.d[i]>g2.d[i])
return 0;
return g1.val;
}
inline LL into(LL now){
if(!now)
return 0;
LL sum=0;
for(LL i=0;i<=3;++i)
if(tree[now].mx[i]<=tmp.d[i])
++sum;
if(sum==4)
return 4;
for(LL i=0;i<=3;++i)
if(tree[now].mi[i]>tmp.d[i])
return 0;
return 1;
}
void query(LL now){
ans=max(dis(tree[now],tmp),ans);
LL son0=tree[now].son[0],son1=tree[now].son[1];
LL dl=into(son0),dr=into(son1);
if(dl==4)
ans=max(tree[son0].maxn,ans);
else if(dl&&ans<tree[son0].maxn)
query(son0);
if(dr==4)
ans=max(tree[son1].maxn,ans);
else if(dr&&ans<tree[son1].maxn)
query(son1);
}
int main(){
n=read();
for(LL i=1;i<=n;++i){
for(LL j=0;j<=3;++j)
t2[i].d[j]=read();
t2[i].val=1;
}
sort(t2+1,t2+n+1);
for(LL i=0;i<=3;++i)
tree[0].mi[i]=inf,tree[0].mx[i]=-inf; tree[0].maxn=-inf;
for(LL i=1;i<=n;++i){
ans=0;
tmp=t2[i];
query(root);
tmp.val+=ans;
ans_maxn=max(tmp.val,ans_maxn);
insert(root,0);
}
printf("%lld",ans_maxn);
return 0;
}
luoguP3769 [CH弱省胡策R2]TATT的更多相关文章
- [CH弱省胡策R2]TATT
description 洛谷 data range \[ n\le 5\times 10^4\] solution 这就是四维偏序了... 好象时间复杂度是\(O(n^{\frac{5}{3}})\) ...
- [Luogu3769][CH弱省胡策R2]TATT
luogu 题意 其实就是四维偏序. sol 第一维排序,然后就只需要写个\(3D-tree\)了. 据说\(kD-tree\)的单次查询复杂度是\(O(n^{1-\frac{1}{k}})\).所以 ...
- 【题解】[CH弱省胡策R2]TATT
本蒟蒻第一道\(K-D-Tree\)维护\(dp\) Question 题目大意:求一条路径,使得其四个维度单调不降. 先排序消掉一维再说. 对于每一个点,初始的时候绝对长度是1啊.于是,先赋值一个1 ...
- 洛谷3769[CH弱省胡策R2]TATT (KDTree)(四维LIS)
真是一个自闭的题目(调了一个上午+大半个下午) 从\(WA\)到\(WA+TLE\)到\(TLE\)到\(AC\) 真的艰辛. 首先,这个题,我们可以考虑直接上四维KDTree来解决. 对于kdtre ...
- 【弱省胡策】Round #5 Count
[弱省胡策]Round #5 Count 太神仙了. \(DP\)做法 设\(f_{n,m,d,k}\)表示\(n*m\)的矩阵,填入第\(k\)个颜色,并且第\(k\)个颜色最少的一列上有\(d\) ...
- 弱省胡策 Magic
弱省胡策 Magic 求\(n\)个点\(n\)的条边的简单联通图的个数. 毒瘤,还要写高精. 我们枚举环的大小\(k\),\(\displaystyle ans=\sum_{k=3}^nC_n^k ...
- 【ContestHunter】【弱省胡策】【Round0】(A)&【Round1】(B)
DP+容斥原理or补集转化?/KD-Tree 唔……突然发现最早打的两场(打的最烂的两场)没有写记录……(太烂所以不忍记录了吗... 还是把搞出来了的两道题记录一下吧= =勉强算弥补一下缺憾…… Ro ...
- 【ContestHunter】【弱省胡策】【Round3】(C)
容斥原理+Fib Orz HE的神犇们 蒟蒻只能改出来第三题……实在太弱 官方题解:http://pan.baidu.com/s/1o6MdtQq fib的神奇性质……还有解密a[i]的过程……这里就 ...
- 【ContestHunter】【弱省胡策】【Round2】
官方题解:http://wyfcyx.is-programmer.com/posts/95490.html A 目前只会30分的暴力……DP好像很神的样子0.0(听说可以多次随机强行算? //Roun ...
随机推荐
- 【前端GUI】—— 网站美工必须掌握的PS知识点&思维导图
前言:前端离不开与设计的沟通,有时候还需要自己上手改动甚至设计网页,所以这里简单梳理一下近期学习的“网站美工”相关知识及练习.(工作用不上的时候,自己玩儿着也蛮有意思的,哈哈(*゚∀゚*)~) 一.P ...
- (Caffe)基本类Blob,Layer,Net(一)
本文地址:http://blog.csdn.net/mounty_fsc/article/details/51085654 Caffe中,Blob.Layer,Net,Solver是最为核心的类,下面 ...
- binary-tree-zigzag-level-order-traversal——二叉树分层输出
Given a binary tree, return the zigzag level order traversal of its nodes' values. (ie, from left to ...
- 【Excle数据透视表】如何创建非共享缓存的数据透视表
一般情况下,利用同一个数据源创建多个数据表时,默认创建的是共享缓存的数据透视表.刷新一个数据透视表时会影响其他数据透视表的展示结果. 解决方案 创建非共享缓存的多个数据透视表 步骤一 单击工作表数据任 ...
- React学习之事件绑定
React事件绑定有主要有三种方式 第一种官方推荐方式: class LoginControl extends React.Component { constructor(props) { ...
- Matlab 绘图全方位分析及源码
Matlab绘图 强大的绘图功能是Matlab的特点之一,Matlab提供了一系列的绘图函数,用户不需要过多的考虑绘图的细节,只需要给出一些基本参数就能得到所需图形,这类函数称为高层绘图函数.此外,M ...
- 关于iOS Tabbar的一些设置
事实上iOS Tabbar的可定制性很高,我们没有必要反复造轮子,以下是笔者收集的一些tabbar的经常使用设置.希望对大家有所帮助. 设置tabbar选中颜色 iOS7设置例如以下: [self.t ...
- QTreeWidget 的用法
Qt QTreeWidget 新建一个Qt Widgets Application,拖拽一个Tree Widget 到 ui 界面上,最后实现的效果如下: 添加代码 //test.h //在头文件里添 ...
- CSS - 修改input - placeholder 和 readonly 的样式
placeholder ::-webkit-input-placeholder { /* WebKit browsers */ color: #999999; } :-moz-placeholder ...
- Controller跳转到外部链接
return new ModelAndView(new RedirectView("http://www.baidu.com")); 用了这个方法之后,方法的返回值也会变成mode ...