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 ...
随机推荐
- MFC中 自定义类访问主对话框控件的方法
之前一直在找有木有好点的方法.现在终于被我找到,收藏之~~~~~~ 在使用mfc的时候经常遇到自定义类访问主对话框控件的问题,例如自定义类中的方法要输出一段字符串到主对话框的EDIT控件.控制对话框的 ...
- websocket关于禁止一个账号多窗口链接的问题
通过websocket的session.getSessionId()与oldSession.getSessionId()来equals判断是否是新窗口. 如果不同不让链接. 问题1.虽然新来的链接连不 ...
- Linux测网速
$ wget https://raw.github.com/sivel/speedtest-cli/master/speedtest_cli.py$ chmod a+rx speedtest_cli. ...
- SQL之相关语法及操作符
概述:UNION.SELECT INTO.INSERT INTO SELECT.SQL约束 UNION操作符 UNION操作符用于合并两个或多个SELECT语句的结果集 请注意,UNION内部的每个S ...
- C#.Net调试时调无法“编辑并继续”
‘启用编辑并继续’的作用是允许用户在调试的过程中修改源代码,并且修改的代码会编译到调试进程中立刻生效. 在调试时,无法查到变量的值,也无法编辑,若编辑会跳出如下弹框: 解决方法: 1:如下图,选择De ...
- C# 请求Web Api 接口,返回的json数据直接反序列化为实体类
须要的引用的dll类: Newtonsoft.Json.dll.System.Net.Http.dll.System.Net.Http.Formatting.dll Web Api接口为GET形式: ...
- Android网络编程Socket【实例解析】
Socket 事实上和JavaWeb 里面的Socket一模一样 建立客服端,server端,server开一个port供客服端訪问 第一步创建server端:(这里把为了便于解说.把server端, ...
- 【强网杯2018】逆向hide
这是事后才做出来的,网上没有找到现成的writeup,所以在这里记录一下 UPX加壳,而且linux下upx -d无法解,也无法gdb/ida attach 因为是64位,所以没有pushad,只能挨 ...
- 自己定义ViewGroup控件(二)----->流式布局进阶(二)
main.xml <?xml version="1.0" encoding="utf-8"? > <com.example.SimpleLay ...
- hashCode与equals的作用与区别及应当注意的细节
最近去面试了几家公司,被问到hashCode的作用,虽然回答出来了,但是自己还是对hashCode和equals的作用一知半解的,所以决定把它们研究一下. 以前写程序一直没有注意hashCode的作用 ...