题目链接

\(Description\)

有n个座山,其高度为hi。有m条带权双向边连接某些山。多次询问,每次询问从v出发 只经过边权<=x的边 所能到达的山中,第K高的是多少。

\(Solution\)

x的限制将图分成了若干连通块,如果x单调递增的话,只需要合并连通块就可以了。

离线,并查集维护所属连通块,线段树合并状态、查询K大。(强制在线的话可以主席树?BZOJ3551 不想写了还是写了,还好)

强制在线的Kruskal+主席树做法:BZOJ3551

//39360kb	5248ms
#include <cstdio>
#include <cctype>
#include <algorithm>
//#define gc() getchar()
#define MAXIN 250000
#define gc() (SS==TT&&(TT=(SS=IN)+fread(IN,1,MAXIN,stdin),SS==TT)?EOF:*SS++)
const int N=1e5+5,M=5e5+5; int n,m,Q,cnt,ref[N],h[N],Ans[M],fa[N],root[N];
char IN[MAXIN],*SS=IN,*TT=IN;
inline int read();
struct Edge{
int fr,to,val;
inline void Read(){
fr=read(), to=read(), val=read();
}
bool operator <(const Edge &a)const{
return val<a.val;
}
}e[M];
struct QUERY{
int fr,x,k,id;
inline void Read(int i){
fr=read(), x=read(), k=read(), id=i;
}
bool operator <(const QUERY &a)const{
return x<a.x;
}
}q[M];
struct Segment_Tree
{
#define S N*18
#define lson son[x][0]
#define rson son[x][1]
int tot,sz[S],son[S][2]; void Insert(int &x,int l,int r,int p)
{
sz[x=++tot]=1;
if(l==r) return;
int m=l+r>>1;
if(p<=m) Insert(lson,l,m,p);
else Insert(rson,m+1,r,p);
}
int Merge(int x,int y)
{
if(!x||!y) return x^y;
lson=Merge(lson,son[y][0]), rson=Merge(rson,son[y][1]);
sz[x]+=sz[y]; /*sz[y]=0;*/ return x;
}
int Query(int x,int l,int r,int k)
{
if(l==r) return l;
int ls=sz[lson];
if(k<=ls) return Query(lson,l,l+r>>1,k);
return Query(rson,(l+r>>1)+1,r,k-ls);
}
}T; inline int read()
{
int now=0,f=1;register char c=gc();
for(;!isdigit(c);c=='-'&&(f=-1),c=gc());
for(;isdigit(c);now=now*10+c-'0',c=gc());
return now*f;
}
int Getfa(int x){
return x==fa[x]?x:fa[x]=Getfa(fa[x]);
}
inline int Find(int x)
{
int l=1,r=cnt,mid;
while(l<r)
if(ref[mid=l+r>>1]<x) l=mid+1;
else r=mid;
return l;
}
void Merge(int x,int y)
{
int r1=Getfa(x), r2=Getfa(y);
if(r1==r2) return;
fa[r2]=r1, root[r1]=T.Merge(root[r1],root[r2]);
}
int Query(int x,int k)
{
int rt=Getfa(x);
return T.sz[root[rt]]<k?-1:ref[T.Query(root[rt],1,cnt,T.sz[root[rt]]-k+1)];
} int main()
{
n=read(), m=read(), Q=read();
for(int i=1; i<=n; ++i) ref[i]=h[i]=read(), fa[i]=i;
std::sort(ref+1,ref+1+n), cnt=1;
for(int i=2; i<=n; ++i) if(ref[i]!=ref[i-1]) ref[++cnt]=ref[i];
for(int i=1; i<=n; ++i) h[i]=Find(h[i]), T.Insert(root[i],1,cnt,h[i]); for(int i=1; i<=m; ++i) e[i].Read();
std::sort(e+1,e+1+m);
for(int i=1; i<=Q; ++i) q[i].Read(i);
std::sort(q+1,q+1+Q); for(int i=1,now=1; i<=Q; ++i)
{
while(now<=m && e[now].val<=q[i].x) Merge(e[now].fr,e[now].to), ++now;
Ans[q[i].id]=Query(q[i].fr,q[i].k);
}
for(int i=1; i<=Q; ++i) printf("%d\n",Ans[i]); return 0;
}

2019.11.15:

//39404kb	5876ms 跑的又比以前慢自闭了
#include <cstdio>
#include <cctype>
#include <algorithm>
#define gc() getchar()
//#define MAXIN 300000
//#define gc() (SS==TT&&(TT=(SS=IN)+fread(IN,1,MAXIN,stdin),SS==TT)?EOF:*SS++)
typedef long long LL;
const int N=1e5+5,M=5e5+5; int ref[N],h[N],Ans[M],root[N],fa[N];
//char IN[MAXIN],*SS=IN,*TT=IN;
struct Edge
{
int u,v,w;
bool operator <(const Edge &x)const
{
return w<x.w;
}
}e[M];
struct Queries
{
int s,v,k,id;
bool operator <(const Queries &x)const
{
return v<x.v;
}
}q[M];
struct Segment_Tree
{
#define S N*18
#define ls son[x][0]
#define rs son[x][1]
#define lson ls,l,l+r>>1
#define rson rs,(l+r>>1)+1,r
int tot,son[S][2],sz[S];
#undef S void Insert(int &x,int l,int r,int p)
{
sz[x=++tot]=1;
if(l<r) p<=(l+r>>1) ? Insert(lson,p) : Insert(rson,p);
}
int Merge(int x,int y)
{
if(!x||!y) return x|y;
ls=Merge(ls,son[y][0]), rs=Merge(rs,son[y][1]);
return sz[x]+=sz[y],x;
}
int Query(int x,int l,int r,int k)
{
if(l==r) return l;
int lsz=sz[ls];
return lsz>=k?Query(lson,k):Query(rson,k-lsz);
}
}T; inline int read()
{
int now=0;register char c=gc();
for(;!isdigit(c);c=gc());
for(;isdigit(c);now=now*10+c-48,c=gc());
return now;
}
int Getfa(int x)
{
return x==fa[x]?x:fa[x]=Getfa(fa[x]);
}
inline int Find(int x,int r)
{
int l=1,mid;
while(l<r)
if(ref[mid=l+r>>1]<x) l=mid+1;
else r=mid;
return l;
}
inline void Merge(int x,int y)
{
int r1=Getfa(x),r2=Getfa(y);
if(r1!=r2) fa[r2]=r1, root[r1]=T.Merge(root[r1],root[r2]);
}
inline int Query(int n,int x,int k)
{
int rt=root[Getfa(x)];
return T.sz[rt]<k?-1:ref[T.Query(rt,1,n,T.sz[rt]-k+1)];
} int main()
{
const int n=read(),m=read(),Q=read();
for(int i=1; i<=n; ++i) ref[i]=h[i]=read();
std::sort(ref+1,ref+1+n); int cnt=1;
for(int i=2; i<=n; ++i) if(ref[i]!=ref[i-1]) ref[++cnt]=ref[i];
for(int i=1; i<=n; ++i) h[i]=Find(h[i],cnt);
for(int i=1; i<=n; ++i) T.Insert(root[i],1,cnt,h[i]); for(int i=1; i<=m; ++i) e[i]=(Edge){read(),read(),read()};
for(int i=1; i<=Q; ++i) q[i]=(Queries){read(),read(),read(),i};
std::sort(e+1,e+1+m), std::sort(q+1,q+1+Q); for(int i=1; i<=n; ++i) fa[i]=i;
e[m+1].w=2e9;
for(int i=1,now=1; i<=Q; ++i)
{
while(e[now].w<=q[i].v) Merge(e[now].u,e[now].v), ++now;
Ans[q[i].id]=Query(cnt,q[i].s,q[i].k);
}
for(int i=1; i<=Q; printf("%d\n",Ans[i++])); return 0;
}

BZOJ.3545.[ONTAK2010]Peaks(线段树合并)的更多相关文章

  1. 【bzoj3545】[ONTAK2010]Peaks 线段树合并

    [bzoj3545][ONTAK2010]Peaks 2014年8月26日3,1512 Description 在Bytemountains有N座山峰,每座山峰有他的高度h_i.有些山峰之间有双向道路 ...

  2. BZOJ 3545: [ONTAK2010]Peaks( BST + 启发式合并 + 并查集 )

    这道题很好想, 离线, 按询问的x排序从小到大, 然后用并查集维护连通性, 用平衡树维护连通块的山的权值, 合并就用启发式合并.时间复杂度的话, 排序是O(mlogm + qlogq), 启发式合并是 ...

  3. BZOJ 3545: [ONTAK2010]Peaks [Splay启发式合并]

    3545: [ONTAK2010]Peaks 题意:带权图,多组询问与一个点通过边权\(\le x\)的边连通的点中点权k大值 又读错题了,输出点一直WA,问的是点权啊 本题加强版强制在线了,那这道题 ...

  4. Peaks 线段树合并

    Peaks 线段树合并 \(n\)个带权值\(h_i\)山峰,有\(m\)条山峰间双向道路,\(q\)组询问,问从\(v_i\)开始只经过\(h_i\le x\)的路径所能到达的山峰中第\(k\)高的 ...

  5. ●BZOJ 3545 [ONTAK2010]Peaks(离线)

    题链: http://www.lydsy.com/JudgeOnline/problem.php?id=3545 http://www.lydsy.com/JudgeOnline/problem.ph ...

  6. bzoj 4631: 踩气球 线段树合并

    4631: 踩气球 Time Limit: 10 Sec  Memory Limit: 256 MBSubmit: 265  Solved: 136[Submit][Status][Discuss] ...

  7. BZOJ:5457: 城市(线段树合并)(尚待优化)

    5457: 城市 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 18  Solved: 12[Submit][Status][Discuss] Des ...

  8. bzoj3545 Peaks 线段树合并

    离线乱搞... 也就是一个线段树合并没什么 #include<algorithm> #include<iostream> #include<cstring> #in ...

  9. bzoj 3545: [ONTAK2010]Peaks

    Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 1124  Solved: 304[Submit][Status][Discuss] Descripti ...

随机推荐

  1. MFC创建线程示例

    MFC创建线程示例 AfxBeginThread() 创建现场的方法是AfxBeginThread()函数. 在[.CPP]文件定义一个全局变量,决定什么时候退出这个线程. BOOL g_bWillE ...

  2. JavaScript中 this 的指向

    很多人都会被JavaScript中this的指向(也就是函数在调用时的调用上下文)弄晕,这里做一下总结: 首先,顶层的this指向全局对象. 函数中的this按照调用方法的不同,其指向也不同: 1.函 ...

  3. python日志和异常

    “日志”转载:http://www.cnblogs.com/dkblog/archive/2011/08/26/2155018.html "异常"转载:http://www.cnb ...

  4. kafka系列一、kafka安装及部署、集群搭建

    一.环境准备 操作系统:Cent OS 7 Kafka版本:kafka_2.10 Kafka官网下载:请点击 JDK版本:1.8.0_171 zookeeper-3.4.10 二.kafka安装配置 ...

  5. 找到多个与名为“Home”的控制器匹配的类型的解决方案

    主地址:http://localhost:3412/Home/Index 区域地址:http://localhost:3412/T200/Home/Index 解决方法: 注册路由添加命名空间(nam ...

  6. Springboot分模块开发详解(1):建立父工程

    基础服务,见下: base是父工程,base-entity是实体层,base-dao是DAO层,base-service是业务层,base-controller是WEB控制器层,base-web是页面 ...

  7. Account的简单架构

    前几天,有园友私下问我,博客中的AccountDemo后端架构为什么是那样的,是不是分层太多太冗余,故这里简单介绍下.先看解决方案工程截图: 每个工程的含义,见https://www.cnblogs. ...

  8. Ex 6_3 修建酒店所获得的利润..._第五次作业

    假设profit[i]为在前i个位置修建酒店所获得的最大利润,当i=0时,profit[0]=0.当i>0时,若j为上一个满足m[i]-m[j]k的位置.若profit[i-1]>prof ...

  9. Oracle常用sql语句(一)

    # Sql的分类 # DDL (Data Definition Language):数据定义语言,用来定义数据库对象:库.表.列等: CREATE. ALTER.DROP DML(Data Manip ...

  10. js闭包实例汇总

    本文是通过实例来帮助大家深刻理解js闭包,是篇非常不错的文章,这里推荐给大家,有需要的小伙伴可以参考下 Js闭包 闭包前要了解的知识 1. 函数作用域 (1).Js语言特殊之处在于函数内部可以直接读取 ...