打开CDQ的大门&BZOJ3262
第一次接触CDQ分治,感谢YZ大佬的教导。
CDQ分治就是一种奇特的分治方法,它用左区间的区间信息来更新右区间。
设CDQ(L,R,l,r)表示递归到区间[L,R],区间的值为[l,r]。
mid=l+r/2。
将L~R区间按<=mid和>mid的两块重新分开。
继续递归区间分治。
CDQ分治用于解决一类偏序问题,题目中所求的为三维偏序。
即计算一类ai<bi,aj<bj,ak<bk的贡献。
普通的二维偏序大家都做过,经典题目逆序对。
三维怎么做呢?
第一维依然是SORT,第三位是树状数组。
第二维是CDQ。
第一维SORT后有序,故无需考虑第一维。
第二维CDQ分治时枚举L~R区间,当a[i].y>mid时,计算在权值树状数组上的贡献,否则插入树状数组。
注意:这道题需要对重复区间判重,否则贡献值不好算。
code:
/**************************************************************
Problem: 3262
User: yekehe
Language: C++
Result: Accepted
Time:1116 ms
Memory:10600 kb
****************************************************************/
#include <cstdio>
#include <algorithm>
using namespace std;
char tc()
{
static char tr[],*A=tr,*B=tr;
return A==B&&(B=(A=tr)+fread(tr,,,stdin),A==B)?EOF:*A++;
}
int read()
{
char c;while(c=tc(),c<''||c>'');
int x=c-'';while(c=tc(),c>=''&&c<='')x=x*+c-'';
return x;
}
struct node{
int x,y,z,o;
}a[],lx[],rx[],Ne[];
int N,K,cnt;
int at[],b[],tim,ans[],tot[];
inline int cmp(node x,node y){return x.x<y.x || x.x==y.x&&x.y<y.y || x.x==y.x&&x.y==y.y&&x.z<y.z;}
void add(int x,int y)
{
for(int i=x;i<=K;i+=i&-i){
if(b[i]!=tim)at[i]=,b[i]=tim;
at[i]+=y;
}
return ;
}
int get(int x)
{
int tot=;
for(int i=x;i;i-=i&-i)
if(!(b[i]^tim))tot+=at[i];
return tot;
}
void cdq(int L,int R,int l,int r)
{
tim++;
int mid=l+r>>,lt=,rt=;
for(int i=L;i<=R;i++){
if(!(l^r)){
ans[Ne[i].o]+=get(Ne[i].z);
add(Ne[i].z,tot[Ne[i].o]);
}
else{
if(Ne[i].y<=mid)add(Ne[i].z,tot[Ne[i].o]);
else ans[Ne[i].o]+=get(Ne[i].z);
}
}//统计贡献
for(int i=L;i<=R;i++){
if(Ne[i].y<=mid)lx[++lt]=Ne[i];
else rx[++rt]=Ne[i];
}
for(int i=;i<=lt;i++)Ne[L+i-]=lx[i];
for(int i=;i<=rt;i++)Ne[L+i-+lt]=rx[i];//类似于归并那样重组
if(!(l^r))return ;
cdq(L,L+lt-,l,mid);
cdq(L+lt,R,mid+,r);
return ;
}
int IS(int x,int y){
return a[x].x==a[y].x&&a[x].y==a[y].y&&a[x].z==a[y].z;
}
int res[];
int main()
{
N=read(),K=read();
for(int i=;i<=N;i++){
a[i].x=read(),a[i].y=read(),a[i].z=read();
}
sort(a+,a+N+,cmp);
a[].x=a[].y=a[].z=-2e9;
for(int i=;i<=N;i++){
if(IS(i,i-))++tot[cnt];
else tot[++cnt]=,Ne[cnt]=a[i],Ne[cnt].o=cnt;
}//去重
cdq(,cnt,,K);//注意为cnt
for(int i=;i<=cnt;i++)res[ans[i]+tot[i]-]+=tot[i];
for(int i=;i<N;i++)printf("%d\n",res[i]);
return ;
}
打开CDQ的大门&BZOJ3262的更多相关文章
- 文科生打开python的大门
作为唯一的一名教育学院的学生,加入python课程,一定要声明我可不是并不是被迫选课的!虽然是文科生,但是是对编程这种东西很感兴趣的文科生.从站在python门口的张望,到现在悄悄把门打开,越来越感觉 ...
- 第一步,怎么打开react的大门?
前言 其实我的react的实战经验很少,大概是17年-18年写了一个react全家桶的后台管理系统.猜测这个项目应该还一直在使用 在我手里的vue项目就比较多了,技术栈一直是vue全家桶.最近也在深入 ...
- 菜鸟VUER学习记——零0章、打开新的大门
是什么 基于MVVM模型,核心库只关注视图层,通过尽可能简单的 API 实现响应的数据绑定和组合的视图组件的js框架.根据项目的复杂度和需求,渐进的加入插件以达到恰到好处的程度. 解决问题 UI组件化 ...
- Python打开新世界的大门-入门篇1
目录 题记 Python技巧.避坑及心得 八种数据类型 循环 函数 Homework 题外话 之前没有写博客的习惯,现在开始写觉得入门也太晚了吧,看看同龄的大哥都写了十几万字.于是 ...
- echart 打开新世界的大门
实时折线图 option = { backgroundColor:'#2B2B2B', tooltip: { trigger: 'axis' }, legend: { data:['频率'], tex ...
- cdq分治入门--BZOJ3262: 陌上花开
n<=100000个人,每个人三个属性Ai,Bi,Ci,一个人i的等级为Ai>=Aj,Bi>=Bj,Ci>=Cj的人数,求每个等级有多少人. 裸的三维偏序.按照常规思路,一维排 ...
- GoLang——Hello World,打开新世界的大门
本文始发于个人公众号:TechFlow,原创不易,求个关注 今天是Go语言系列的第一篇文章,我们来聊聊这门新的语言和它的基础语法. 浅谈Golang 作为程序员而言,往往对于学习新的语言都是有抗拒的. ...
- bzoj 2163: 复杂的大门
2163: 复杂的大门 Time Limit: 20 Sec Memory Limit: 259 MBSubmit: 418 Solved: 259[Submit][Status][Discuss ...
- BZOJ2163: 复杂的大门
BZOJ2163: 复杂的大门 Description 你去找某bm玩,到了门口才发现要打开他家的大门不是一件容易的事……他家的大门外有n个站台,用1到n的正整数编号.你需要对每个站台访问一定次数以后 ...
随机推荐
- SQL Server 索引知识-应用,维护
创建聚集索引 a索引键最好唯一(如果不唯一会隐形建立uniquier列(4字节)确保唯一,也就是这列都会复制到所有非聚集索引中) b聚集索引列所占空间应尽量小(否则也会使非聚集索引的空间变大) c聚集 ...
- 解决国外模板h1、h2、h3...不显示中文文章标题的问题
如果你经常用国外好看的网页模版时候,会遇到不显示中文文章标题的情况,显示英文标题却正常.遇到这个情况很多人认为应该修改CSS的font-family的字体,其实这是错误的,与CSS无关. 出现这种情况 ...
- golang 防知乎 中文验证码 源码
原创,转载请注明出处! 最开始用图形来模仿文字进行各种角度的倒立和排列,后来切换为文字后,有很多问题.总结如下: 1.程序在画图形和画文字方面不一样,图形的是从原点开始(0,0),而文字则从文字的基线 ...
- IP地址分类及CIDR划分方法
IP地址的分类和表示有三种形式,1.分类的IP地址.2.子网划分.3.无分类编址CIDR 1.分类的IP地址 IP地址:: = {<网络号>,<主机号>} 不同的网络号和主机号 ...
- webshell扫描
可扫描 weevelyshell 生成 或加密的shell 及各种变异webshell 目前仅支持php 支持扫描 weevelyshell 生成 或加密的shell 支持扫描callback一句话s ...
- UI(一)
1.AfxWinMain 首先,MFC程序先执行到TheApp实例化对象也就是通过这句CTestApp the App来实例化对象的然后,调用CTestApp构造函数分配内存空间 然后,就调用了Afx ...
- 跟我一起阅读Java源代码之HashMap(三)
上一节我们讲到了如何用散列和链表实现HashMap,其中有一个疑问今天已经有些答案了,为什么要用链表而不是数组 链表的作用有如下两点好处 1. remove操作时效率高,只维护指针的变化即可,无需进行 ...
- IKVM.NET入门(2)
ikvm.net是什么 http://www.ikvm.net/ ikvm.net是能够运行在mono和.net framework的java虚拟机.它包括了 在.net中实现的一个java虚拟机 j ...
- Python实现向s3共享存储上传和下载文件
#!/usr/bin/env python #-*- encoding: utf8 -*- import boto import boto.s3.connection from boto.s3.key ...
- debian 7上安装svn
1.在终端中直接输入 sudo apt-get install subversion,选择安装即可 2.查看版本命令 svnserve --version(更多命令直接键入svnserve --he ...