题目链接

\(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. 【转】Python之数据序列化(json、pickle、shelve)

    [转]Python之数据序列化(json.pickle.shelve) 本节内容 前言 json模块 pickle模块 shelve模块 总结 一.前言 1. 现实需求 每种编程语言都有各自的数据类型 ...

  2. Python Tools for Machine Learning

    Python Tools for Machine Learning Python is one of the best programming languages out there, with an ...

  3. vim 超强发行版

    推荐第一个: https://github.com/spf13/spf13-vim https://github.com/Spacevim/Spacevim https://github.com/JB ...

  4. API 开发平台 dreamfactory,参考SAWAGGER,国外厂家,开源,本地与云部署

    API 开发平台,参考SAWAGGER,国外厂家,本地与云部署:参考  http://swagger.io/commercial-tools/ 1.dreamfactory 梦工厂公司  https: ...

  5. 如何在windows上调试安卓机谷歌浏览器上的页面

    - 下面的方法仅在windows和安卓机上测试过,,,, - 手机(安卓机)需要安装chrome与电脑(Windows)上的chrome配合,也就是只能调试谷歌浏览器上的页面 1.手机的准备工作 打开 ...

  6. PNG,JPEG,BMP,JIF图片格式详解及其对比

    原文地址:http://blog.csdn.net/u012611878/article/details/52215985 图片格式详解 不知道大家有没有注意过网页里,手机里,平板里的图片,事实上,图 ...

  7. mybatis和spring整合的关键配置

    spring配置文件 applicationContext.xml: <beans xmlns="http://www.springframework.org/schema/beans ...

  8. 使用FreeSWITCH做电话自动回访设置

    一.背景介绍: 目前公司在处理客户回访方面,需要人工进行电话回访,尤其是逢年过节的时候,电话问候更能体现服务的品质: 在某些公司,电话销售员需要给大批量的陌生用户打电话,如果能过滤掉不关心的用户,销售 ...

  9. Android中PopupWindow用法

    参考资料链接:http://developer.android.com/reference/android/widget/PopupWindow.html 在Android中有很多级别的Window, ...

  10. Day6-------------ext4文件系统

    1.cp /etc/passwd /sdb6 把/etc/passwd的内容写入 sdb6 写入过程:日志------------>刷到硬盘 2.ext4已经有点过时 xfs:可存海量数据 bt ...