[BZOJ 3262]陌上开花
今天写了到偏序问题,发现博主真的是个傻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]陌上开花的更多相关文章
- bzoj 3262 陌上花开 - CDQ分治 - 树状数组
Description 有n朵花,每朵花有三个属性:花形(s).颜色(c).气味(m),又三个整数表示.现要对每朵花评级,一朵花的级别是它拥有的美丽能超过的花的数量.定义一朵花A比另一朵花B要美丽,当 ...
- BZOJ.3262.陌上花开([模板]CDQ分治 三维偏序)
题目链接 BZOJ3262 洛谷P3810 /* 5904kb 872ms 对于相邻x,y,z相同的元素要进行去重,并记录次数算入贡献(它们之间产生的答案是一样的,但不去重会..) */ #inclu ...
- Luogu 3810 & BZOJ 3262 陌上花开/三维偏序 | CDQ分治
Luogu 3810 & BZOJ 3263 陌上花开/三维偏序 | CDQ分治 题面 \(n\)个元素,每个元素有三个值:\(a_i\), \(b_i\) 和 \(c_i\).定义一个元素的 ...
- 【BZOJ 3262】 3262: 陌上花开 (CDQ分治)
3262: 陌上花开 Description 有n朵花,每朵花有三个属性:花形(s).颜色(c).气味(m),又三个整数表示.现要对每朵花评级,一朵花的级别是它拥有的美丽能超过的花的数量.定义一朵花A ...
- BZOJ 3262 陌上花开 ——CDQ分治
[题目分析] 多维问题,我们可以按照其中一维排序,然后把这一维抽象的改为时间. 然后剩下两维,就像简单题那样,排序一维,树状数组一维,按照时间分治即可. 挺有套路的一种算法. 时间的抽象很巧妙. 同种 ...
- bzoj 3262 陌上花开
本质是一个三维偏序,一位排序后cdq分治,一维在子函数里排序,一维用树状数组维护. 把三维相等的合并到一个里面. #include<iostream> #include<cstdio ...
- BZOJ 3262 陌上花开 CDQ分治
= =原来复杂度还是nlog^2(n) Orz 被喷了 #include<cstdio> #include<cstdlib> #include<algorithm> ...
- BZOJ 3262: 陌上花开 [CDQ分治 三维偏序]
Description 有n朵花,每朵花有三个属性:花形(s).颜色(c).气味(m),又三个整数表示.现要对每朵花评级,一朵花的级别是它拥有的美丽能超过的花的数量.定义一朵花A比另一朵花B要美丽,当 ...
- 【刷题】BZOJ 3262 [HNOI2008]GT考试
Description 阿申准备报名参加GT考试,准考证号为N位数X1X2....Xn(0<=Xi<=9),他不希望准考证号上出现不吉利的数字. 他的不吉利数学A1A2...Am(0< ...
随机推荐
- opennebula 一些问t题讨论
ou_ian - June 8th, 2011 11:13 am非常想请教一下如何设置VM的ip为DHCP? 我们的VM都需要通过eth0 (通过公司的DHCP获得ip)来连接到公司的网络(Inter ...
- PHP+SOCKET 模拟HTTP请求
HTTP消息结构 客户端请求包括四部份:请求行(状态行).请求头.空行.请求主体(数据),如下图: 服务端响应包括四部份:响应行(状态行).响应头.空行.响应主体(数据),如图: HTTP请求方法: ...
- 手打的table
突然觉得,如果我不上传源码和写篇博客,对不起花在这个破网页2个小时的时间,完全手打,浏览器调效果. 源码如下: a.html: <!DOCTYPE html PUBLIC "-//W3 ...
- php系统常量
(1)__FILE__ :php程序文件名.它可以帮助我们获取当前文件在服务器的物理位置. (2)__LINE__ :PHP程序文件行数.它可以告诉我们,当前代码在第几行. (3)PHP_VERSIO ...
- 我的linux环境
apache2+php+mysql sudo apt-get install apache2 sudo apt-get install libapache2-mod-php5 php5 sudo ap ...
- web大文件上传控件-设置附加参数-Xproer.HttpUploader6
自定义附加字段在up6.js中定义,也可以不用定义: 注意: 1.附加字段必须是字符串类型. 2.如果附加字段的值包含中文,在上传前必须使用encodeURIComponent进行编码. 在引 ...
- C#NPOI.RabbitMQ.EF.Attribute.HttpRuntime.Cache.AD域.List<T>根据指定字段去重.前端JQuery.Cache.I18N(多语言).data-xx(自定义属性)
使用NPOI 操作Excel 个人使用的电脑基本默认安装Excel 操作起来 调用Excel的组件便可.如果是一台服务器.没有安装Excel,也就无法调用Excel组件. 在此推荐第三方插件.NPOI ...
- JavaScript中事件冒泡之实例理解
此#btnComfirmChooseCompany是Bootstrap模态弹层上的按钮,但点击后,点击事件被Bootstrap外层监听到了, 效果就是模态弹出层被关闭了,所以,我不想这个点击事件被&q ...
- 所谓IIS未注册引起的故障及解决
- 一文读懂spring boot 和微服务的关系
欢迎访问网易云社区,了解更多网易技术产品运营经验. Spring Boot 和微服务没关系, Java 微服务治理框架普遍用的是 Spring Cloud. Spring Boot 产生的背景,是开发 ...