今天写了到偏序问题,发现博主真的是个傻X

传送门

以前的写法

/**************************************************************
    Problem: 3262
    User: MiEcoku
    Language: C++
    Result: Accepted
    Time:3416 ms
    Memory:6296 kb
****************************************************************/

#include <cstdio>
#include <algorithm>
using namespace std;
#define mid ( l + r >> 1)
#define lowbit(x) ( x & -x)

struct GG {
    int x, y, z, id, w;
    void get () {
        scanf("%d%d%d", &x, &y, &z);
    }
}a[], b[];

], k;
; ; x -= lowbit(x)) ans += dis[x]; return ans; }
void add(int x, int y) { for ( ; x <= k; x += lowbit(x)) dis[x] += y; }

bool cmp1(GG a, GG b) { return a.x < b.x || (a.x == b.x && a.y < b.y) || (a.x == b.x && a.y == b.y && a.z < b.z); }
bool cmp2(GG a, GG b) { return a.y < b.y || (a.y == b.y && a.z < b.z); }

];
void cdq(int l, int r) {
    , r);
    sort(b+l, b+mid+, cmp2); sort(b+mid+, b+r+, cmp2);
    ;
    for ( int i = l; i <= mid; i ++) {
        while( b[q].y < b[i].y && q <= r) ans[b[q].id] += query(b[q].z), q ++;
        add(b[i].z, b[i].w);
    }
    while( q <= r) ans[b[q].id] += query(b[q].z), q ++;
    for ( int i = l; i <= mid; i ++) add(b[i].z, -b[i].w);
}

];
int main() {
    scanf("%d%d", &_, &k);
    ; i <= _; i ++) a[i].get();
    sort( a+, a++_, cmp1);
    ; i <= _; i ++) {
        cnt ++;
        ].x || a[i].y != a[i+].y || a[i].z != a[i+].z)
        b[++ n] = a[i], b[n].w = cnt, cnt = , b[n].id = n;
    }
    cdq(, n);
    ; i <= n; i ++)
        d[ans[b[i].id]+b[i].w-] += b[i].w;
    ; i < _; i ++) printf("%d\n", d[i]);
}

可以看出这是个n * logn^2 的算法,这也是博主在大多数地方看到的写法,结果傻叉的以为cdq就是n * long n ^ 2 的算法

实际上可不可以更快?

void cdq(int l, int r) {
    , r);
    sort(b+l, b+mid+, cmp2); sort(b+mid+, b+r+, cmp2);
    ;
    for ( int i = l; i <= mid; i ++) {
        while( b[q].y < b[i].y && q <= r) ans[b[q].id] += query(b[q].z), q ++;
        add(b[i].z, b[i].w);
    }
    while( q <= r) ans[b[q].id] += query(b[q].z), q ++;
    for ( int i = l; i <= mid; i ++) add(b[i].z, -b[i].w);
}

可以看出,在cdq内部排了个序,然而真的有这个必要吗? 答案是否定的

我们都知道归并排序,可以看出,内部这个排序无非是想要将第二位排序

但在内部这个循环内就已经排序,所以那个排序可以省略

我们新开个数组

void cdq(int l, int r) {
    , r);
    , cnt = ;
    for ( int i = l; i <= mid; i ++) {
        while( b[q].y < b[i].y && q <= r) ans[b[q].id] += query(b[q].z), T[++ cnt] = b[q], q ++;
        add(b[i].z, b[i].w); T[++ cnt] = b[i];
    }
    while( q <= r) ans[b[q].id] += query(b[q].z), T[++ cnt] = b[q], q ++;
    for ( int i = l; i <= mid; i ++) add(b[i].z, -b[i].w);
    cnt = ;
    for ( int i = l; i <= r; ++ i) b[i] = T[++ cnt];
}

这样,我们就省略了内部的一个排序问题,做到n * long n

还能再快不?

能!

    for ( int i = l; i <= mid; i ++) add(b[i].z, -b[i].w);

对于这句话,我们能不能将它优化掉?

我们考虑加入一个时间戳 tim

int query(int x, int K) {
    ;
    ; x -= lowbit(x)) if( mark[x] == K) ans += dis[x];
    return ans;
}
void add(int x, int y, int K) {
    for ( ; x <= k; x += lowbit(x)) if( mark[x] == K) dis[x] += y;
    else mark[x] = K, dis[x] = y;
}

然后对树状数组进行点修改

这样我们就将常数再次优化了下

上总代码

/**************************************************************
    Problem: 3262
    User: MiEcoku
    Language: C++
    Result: Accepted
    Time:1428 ms
    Memory:9028 kb
****************************************************************/

/**************************************************************
    Problem: 3262
    User: MiEcoku
    Language: C++
    Result: Accepted
    Time:3416 ms
    Memory:6296 kb
****************************************************************/

#include <cstdio>
#include <algorithm>
using namespace std;
#define mid ( l + r >> 1)
#define lowbit(x) ( x & -x)

struct GG {
    int x, y, z, id, w;
    void get () {
        scanf("%d%d%d", &x, &y, &z);
    }
}a[], b[], T[];

], k, tim, mark[];
int query(int x, int K) {
    ;
    ; x -= lowbit(x)) if( mark[x] == K) ans += dis[x];
    return ans;
}
void add(int x, int y, int K) {
    for ( ; x <= k; x += lowbit(x)) if( mark[x] == K) dis[x] += y;
    else mark[x] = K, dis[x] = y;
}

bool cmp1(GG a, GG b) { return a.x < b.x || (a.x == b.x && a.y < b.y) || (a.x == b.x && a.y == b.y && a.z < b.z); }
bool cmp2(GG a, GG b) { return a.y < b.y || (a.y == b.y && a.z < b.z); }

];
void cdq(int l, int r) {
    , r);
    , cnt = ; ++ tim;
    for ( int i = l; i <= mid; i ++) {
        while( b[q].y < b[i].y && q <= r) ans[b[q].id] += query(b[q].z, tim), T[++ cnt] = b[q], q ++;
        add(b[i].z, b[i].w, tim); T[++ cnt] = b[i];
    }
    while( q <= r) ans[b[q].id] += query(b[q].z, tim), T[++ cnt] = b[q], q ++;
    cnt = ;
    for ( int i = l; i <= r; ++ i) b[i] = T[++ cnt];
}

];
int main() {
    scanf("%d%d", &_, &k);
    ; i <= _; i ++) a[i].get();
    sort( a+, a++_, cmp1);
    ; i <= _; i ++) {
        cnt ++;
        ].x || a[i].y != a[i+].y || a[i].z != a[i+].z)
        b[++ n] = a[i], b[n].w = cnt, cnt = , b[n].id = n;
    }
    cdq(, n);
    ; i <= n; i ++)
        d[ans[b[i].id]+b[i].w-] += b[i].w;
    ; i < _; i ++) printf("%d\n", d[i]);
}

博主因为傻逼不知道这么写然后考试写的KT-tree被卡了


[BZOJ 3262]陌上开花的更多相关文章

  1. bzoj 3262 陌上花开 - CDQ分治 - 树状数组

    Description 有n朵花,每朵花有三个属性:花形(s).颜色(c).气味(m),又三个整数表示.现要对每朵花评级,一朵花的级别是它拥有的美丽能超过的花的数量.定义一朵花A比另一朵花B要美丽,当 ...

  2. BZOJ.3262.陌上花开([模板]CDQ分治 三维偏序)

    题目链接 BZOJ3262 洛谷P3810 /* 5904kb 872ms 对于相邻x,y,z相同的元素要进行去重,并记录次数算入贡献(它们之间产生的答案是一样的,但不去重会..) */ #inclu ...

  3. Luogu 3810 & BZOJ 3262 陌上花开/三维偏序 | CDQ分治

    Luogu 3810 & BZOJ 3263 陌上花开/三维偏序 | CDQ分治 题面 \(n\)个元素,每个元素有三个值:\(a_i\), \(b_i\) 和 \(c_i\).定义一个元素的 ...

  4. 【BZOJ 3262】 3262: 陌上花开 (CDQ分治)

    3262: 陌上花开 Description 有n朵花,每朵花有三个属性:花形(s).颜色(c).气味(m),又三个整数表示.现要对每朵花评级,一朵花的级别是它拥有的美丽能超过的花的数量.定义一朵花A ...

  5. BZOJ 3262 陌上花开 ——CDQ分治

    [题目分析] 多维问题,我们可以按照其中一维排序,然后把这一维抽象的改为时间. 然后剩下两维,就像简单题那样,排序一维,树状数组一维,按照时间分治即可. 挺有套路的一种算法. 时间的抽象很巧妙. 同种 ...

  6. bzoj 3262 陌上花开

    本质是一个三维偏序,一位排序后cdq分治,一维在子函数里排序,一维用树状数组维护. 把三维相等的合并到一个里面. #include<iostream> #include<cstdio ...

  7. BZOJ 3262 陌上花开 CDQ分治

    = =原来复杂度还是nlog^2(n) Orz 被喷了 #include<cstdio> #include<cstdlib> #include<algorithm> ...

  8. BZOJ 3262: 陌上花开 [CDQ分治 三维偏序]

    Description 有n朵花,每朵花有三个属性:花形(s).颜色(c).气味(m),又三个整数表示.现要对每朵花评级,一朵花的级别是它拥有的美丽能超过的花的数量.定义一朵花A比另一朵花B要美丽,当 ...

  9. 【刷题】BZOJ 3262 [HNOI2008]GT考试

    Description 阿申准备报名参加GT考试,准考证号为N位数X1X2....Xn(0<=Xi<=9),他不希望准考证号上出现不吉利的数字. 他的不吉利数学A1A2...Am(0< ...

随机推荐

  1. day58-activiti 02-历史数据查询

    Activity 笔记  第二天 今天内容安排: 1.历史数据查询 办过多少个任务, 这些历史数据有时候我们也需要去查询一下. 本身day02这个项目就没有导jar包,有点类似于maven,在你的项目 ...

  2. C. Ray Tracing——披着搜索外衣的扩展欧几里得

    [题目大意] 给你一个n*m的矩形,光线从(0,0)出发,沿右上方向以每秒根号2米的速度运动,碰到矩形边界就会反弹(符合物理规律的反弹),询问k个点,这些点都在矩形内部且不在矩形边界上,求光经过这些点 ...

  3. jsp Ajax请求(返回xml数据类型)

    <%@ page language="java" import="java.util.*" pageEncoding="UTF-8"% ...

  4. vmware workstation + kvm 部署

    1.物理机BIOS设置里开启虚拟化功能 2.vm里面开启一下两个功能 3.找到存放该虚拟机的dir,找到以.vmx结尾的文件,增加一行 apic.xapic.enabled = "FALSE ...

  5. Linux mii-tool命令

    一.简介 mii-tool 是一个用来查看,管理介质的网络接口的状态的工具. 二.选项 usage: mii-tool [-VvRrwl] [-A media,... | -F media] [int ...

  6. selenium+phantomjs解析JS

    背景知识: PhantomJS 是一个基于WebKit的服务器端 JavaScript API.它全面支持web而不需浏览器支持,其快速,原生支持各种Web标准: DOM 处理, CSS 选择器, J ...

  7. js-day

    1.克莱托指数 公式 :体重(kg) / (身高(m) * 身高(m)) < 20 : 偏瘦 > 20 <25 : 正常 > 25 : 偏旁 步骤: 1.输入体重(weight ...

  8. 关于在审查元素中看到的::before与::after

    审查元素中看到的这两个标签,表示内容并不在元素中,而是在css中,可以查看style看到具体内容. 一般来说这样做是为了清除浮动(clearfix)的代码,防止后边的容器因为浮动出现布局的混乱. 添加 ...

  9. 模板方法(Template Method)模式

    /* * 抽象模版(AbstractClass)角色有如下的责任: 定义了一个或多个抽象操作,以便让子类实现.这些抽象操作叫做基本操作,它们是一个顶级逻辑的组成步骤. 定义并实现了一个模版方法.这个模 ...

  10. C# 高斯消元项目运用

    C# 高斯消元项目运用 最近项目涉及到一个需求,需要把指定数量的多个商品,混合装入到多个不同型号的箱子中(每种型号的箱子装入商品的种类和个数是固定的).这就涉及到解多元一次方程 针对多元一次方程一般用 ...