题意

三维严格偏序最长链。(\(n \le 50000\))

分析

按第一维排序然后以第二和第三维作为关键字依次加入一个二维平面,维护前缀矩形最大值。

题解

当然可以树套树....可是似乎没有随机化算法快..

于是我们上加了强剪枝的kdtree....kdtree大法好...

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N=50105;
struct node;
int A, P, Te=1, top;
struct ip {
int p[3], d;
node *ptr;
void scan() {
for(int i=0; i<3; ++i) {
Te=(ll)A*Te%P;
p[i]=Te;
}
sort(p, p+3);
swap(p[0], p[1]);
swap(p[1], p[2]);
d=0; ptr=0;
}
}a[N], *b[N];
struct node *null;
struct node {
node *c[2], *f;
int x[2], y[2], mx;
ip *d;
void up() {
x[0]=min(d->p[0], min(c[0]->x[0], c[1]->x[0]));
x[1]=max(d->p[0], max(c[0]->x[1], c[1]->x[1]));
y[0]=min(d->p[1], min(c[0]->y[0], c[1]->y[0]));
y[1]=max(d->p[1], max(c[0]->y[1], c[1]->y[1]));
}
void up2() {
mx=max(d->d, max(c[0]->mx, c[1]->mx));
}
bool check(ip *a, int k) {
return x[k]<a->p[0] && y[k]<a->p[1];
}
bool ok(ip *a) {
return d->p[0]<a->p[0] && d->p[1]<a->p[1];
}
void init(ip *a) {
c[0]=c[1]=null;
x[0]=x[1]=a->p[0];
y[0]=y[1]=a->p[1];
mx=0;
d=a;
}
}Po[N], *iT=Po, *root, *st[N];
node *newnode(ip *a) {
iT->init(a);
return iT++;
}
int nowDep;
bool cmp(const ip *x, const ip *y) {
return x->p[nowDep]==y->p[nowDep]?x->p[nowDep^1]<y->p[nowDep^1]:x->p[nowDep]<y->p[nowDep];
}
void build(int l, int r, node *&x, int dep) {
if(l>r) {
x=null;
return;
}
nowDep=dep;
int mid=(l+r)>>1;
nth_element(b+l, b+mid, b+1+r, cmp);
x=newnode(b[mid]);
b[mid]->ptr=x;
build(l, mid-1, x->c[0], dep^1); if(x->c[0]!=null) x->c[0]->f=x;
build(mid+1, r, x->c[1], dep^1); if(x->c[1]!=null) x->c[1]->f=x;
x->up();
}
int askMx;
void ask(ip *a, node *x) {
if(x==null || !x->check(a, 0) || x->mx<=askMx) {
return;
}
if(x->check(a, 1)) {
askMx=max(askMx, x->mx);
}
if(x->ok(a)) askMx=max(askMx, x->d->d);
ask(a, x->c[0]);
ask(a, x->c[1]);
}
void update(node *x) {
for(x->up2(); x!=root; x=x->f, x->up2());
}
int n;
int main() {
null=iT++;
null->x[0]=null->y[0]=~0u>>1;
null->x[1]=null->y[1]=-(~0u>>1);
null->d=0; null->mx=0;
scanf("%d%d%d", &A, &P, &n);
for(int i=1; i<=n; ++i) {
a[i].scan();
b[i]=&a[i];
}
build(1, n, root, 0);
int ans=0;
nowDep=2;
sort(b+1, b+1+n, cmp);
for(int i=1; i<=n; ++i) {
askMx=0;
ask(b[i], root);
b[i]->d=askMx+1;
st[++top]=b[i]->ptr;
if(i!=n && b[i]->p[2]!=b[i+1]->p[2]) {
while(top) update(st[top--]);
}
ans=max(ans, b[i]->d);
}
printf("%d\n", ans);
return 0;
}

【BZOJ】2253: [2010 Beijing wc]纸箱堆叠的更多相关文章

  1. BZOJ 2253: [2010 Beijing wc]纸箱堆叠

    题目 2253: [2010 Beijing wc]纸箱堆叠 Time Limit: 30 Sec  Memory Limit: 256 MBSubmit: 239  Solved: 94 Descr ...

  2. BZOJ2253: [2010 Beijing wc]纸箱堆叠

    题解: 其实就是求三维偏序最长链.类似于三维逆序对,我们可以用树状数组套平衡树来实现. DP方程 :f[i]=max(f[j]+1) a[j]<a[i] 我们按一维排序,另一位建立树状数组,把第 ...

  3. 【BZOJ2253】[2010 Beijing wc]纸箱堆叠 cdq分治

    [BZOJ2253][2010 Beijing wc]纸箱堆叠 Description P 工厂是一个生产纸箱的工厂.纸箱生产线在人工输入三个参数 n p a , , 之后,即可自动化生产三边边长为 ...

  4. BZOJ_2253_[2010 Beijing wc]纸箱堆叠 _CDQ分治+树状数组

    BZOJ_2253_[2010 Beijing wc]纸箱堆叠 _CDQ分治+树状数组 Description P 工厂是一个生产纸箱的工厂.纸箱生产线在人工输入三个参数 n p a , , 之后, ...

  5. BZOJ2253 2010 Beijing wc 纸箱堆叠 CDQ分治

    这题之前度娘上没有CDQ分治做法,gerwYY出来以后写了一个.不过要sort3遍,常数很大. gerw说可以类似划分树的思想优化复杂度,但是蒟蒻目前不会划分树(会了主席树就懒得去弄了). 嗯 将me ...

  6. bzoj 2252 [ 2010 Beijing wc ] 矩阵距离 —— 多源bfs

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2252 又没能自己想出来... 一直在想如何从每个1开始广搜更新答案,再剪剪枝,什么遇到1就不 ...

  7. 【BZOJ】【2253】【WC 2010 BeijingWC】纸箱堆叠

    树套树 Orz zyf 我的树套树不知道为啥一直WA……只好copy了zyf的写法TAT 这题还可以用CDQ分治来做……但是蒟蒻不会…… //y坐标的树状数组是按权值建的……所以需要离散化…… /** ...

  8. 纸箱堆叠 bzoj 2253

    纸箱堆叠 (1s 128MB) box [问题描述] P 工厂是一个生产纸箱的工厂.纸箱生产线在人工输入三个参数 n, p, a 之后,即可自动化生产三边边长为 (a mod P, a^2 mod p ...

  9. 【BZOJ2253】纸箱堆叠 [CDQ分治]

    纸箱堆叠 Time Limit: 30 Sec  Memory Limit: 256 MB[Submit][Status][Discuss] Description P 工厂是一个生产纸箱的工厂. 纸 ...

随机推荐

  1. C#dynamic关键字(1)

    一.object,var,dynamic的区别 static void Main() { //var是C# 3中引入的,其实它仅仅只是一个语法糖. var本身并不是一种类型, 其它两者object和d ...

  2. 并发中的Native方法,CAS操作与ABA问题

    Native方法,Unsafe与CAS操作 >>JNI和Native方法 Java中,通过JNI(Java Native Interface,java本地接口)来实现本地化,访问操作系统底 ...

  3. ArchLinux 安装笔记 --zz

    为何安装 ArchLinux 为了更深层次的理解 Linux (其实只是闲的蛋疼 准备安装介质 U盘首选,没有之一.自己的本子是 MBR 的,UEFI 神马的我才不知道呢哼! 制作 U 盘启动: Li ...

  4. Android N 新特性 + APP开发注意事项

    1. 多窗口MultiWindow 多窗口MultiWindow,这是Android N里对开发者影响比较大的特性,也是大家疑问比较多的地方.站在开发者的角度其实不必太担心这个特性会导致我们需要修改很 ...

  5. 在SQL里如何写条件逻辑?

    主要涉及CASE,WHEN之类.. 不同的服务器上实现if...else...是不一样的. 建议用CASE ,WHEN,因为它们是SQL国标呢. mysql> SELECT -> SUM( ...

  6. java创建线程的几种方式

    1.继承Thread类 /** * @author Ash * @date: 2016年8月6日 下午10:56:45 * @func: 通过继承Thread类来实现多线程 * @email 4086 ...

  7. CodeForces 371D Vessels(树状数组)

    树状数组,一个想法是当往p注水时,认为是其容量变小了,更新时二分枚举,注意一些优化. #include<cstdio> #include<iostream> #include& ...

  8. struts拦截器实现原理

    图1: 上1来源于Struts2官方站点,是Struts 2 的整体结构. 一个请求在Struts2框架中的处理大概分为以下几个步骤 1 客户端初始化一个指向Servlet容器(例如Tomcat)的请 ...

  9. hdu 3032 sg打表找规律 *

    有n堆石子,alice先取,每次可以选择拿走一堆石子中的1~x(该堆石子总数) ,也可以选择将这堆石子分成任意的两堆.alice与bob轮流取,取走最后一个石子的人胜利. 打表代码: #include ...

  10. 课堂随笔 set (集合)

    1.什么是集合:set   (集合)为无序不重复的序列. 2.如何创建一个集合:(1)set()  这样就创建了一个空的集合(2)s1={11,22,33}这样也创建了一个集合.(3)s2=set([ ...