bzoj3551
3551: [ONTAK2010]Peaks加强版
Time Limit: 20 Sec Memory Limit: 128 MB
Submit: 877 Solved: 297
[Submit][Status][Discuss]
Description
Input
Output
Sample Input
Sample Output
HINT
Source
题解:
过了这么久才又开始写博客(省选被虐成渣,现在还要准备半期考试 only shit)好吧,算了毕竟太渣,不多说了,写题解QAQ
看了题目,一脸懵逼,怎么写?(不会) 怎么办?(看题解)
首先看看这道题,它是一个无向图,根本不知道怎么来写主席树,操蛋。
仔细看看题,我们可以这样想,我要走一个边权比x小的边,并且一直走下去,我们不就可以用最小生成树吗?每次走最小的边,那么一定是最优的,所以我们就用kruskal来写
但是要注意的是,在找到一个新的边时,新建一个节点,权值为边的权值,把边的两端连向这个点,不难证明这是一课树。。。。。
然后每次询问就从当前点往上面倍增地跳,直到不能走(点权大于x) 那么那个点的子树中的点就是我可以到达的点
于是题目就变成求一个子树第k大的问题。。。用lca加和主席树就A了
#include<iostream>
#include<cstring>
#include<cstdio>
#include<algorithm>
#include<cmath>
#define maxn 300005
#define maxnode 2000005
using namespace std;
int n,m,q,lastans,top;
int size,tot,ext;
int pre[maxn],now[maxn],v[maxn],fa[maxn],sum[maxnode],son[maxnode][],pp[maxn];
int bin[],deep[maxn],f[maxn][],mx[maxn][];
int h[maxn],list[maxn],qz[maxn],root[maxn],st[maxn],ed[maxn];
bool vis[maxn];
struct date{int u,v,val;
}a[];
int read()
{
int x=; char ch; bool bo=;
while (ch=getchar(),ch<''||ch>'') if (bo=='-') bo=;
while (x=x*+ch-'',ch=getchar(),ch>=''&&ch<='');
if (bo) return -x; return x;
}
bool cmp_val(date a, date b)
{
return a.val<b.val;
}
int find(int x)
{
if (fa[x]!=x) fa[x]=find(fa[x]);
return fa[x];
}
void insert(int x,int y){tot++; pre[tot]=now[x]; now[x]=tot; v[tot]=y;
}
void ins(int l,int r,int x,int &y,int val)
{
int t;
if (!y) y=++tot; sum[y]=sum[x]+;
if (l==r) return;
int mid=(l+r)>>;
if (val<=mid) t=,r=mid; else t=,l=mid+;
son[y][t^]=son[x][t^];
ins(l,r,son[x][t],son[y][t],val);
}
void dfs(int x)
{
vis[x]=; pp[++top]=x;
for (int i=; i<=; i++)
if (bin[i]<=deep[x])
{
f[x][i]=f[f[x][i-]][i-];
mx[x][i]=max(mx[x][i-],mx[f[x][i-]][i-]);
}
for (int p=now[x]; p; p=pre[p])
{
int son=v[p];
deep[son]=deep[x]+;
f[son][]=x;
mx[son][]=qz[x];
dfs(son);
}
if (x>n) pp[++top]=x;
}
void build()
{
ext=n;
sort(a+,a+m+,cmp_val);
for (int i=; i<=m; i++)
{
int q=find(a[i].u),p=find(a[i].v);
if (q!=p)
{
ext++;
fa[q]=fa[p]=ext; qz[ext]=a[i].val;
insert(ext,p); insert(ext,q);
if (ext==*n-) break;
}
}
for (int i=; i<=n; i++)
{
if (!vis[i]) dfs(find(i));
}
for (int i=; i<=top; i++)
{
int t=pp[i];
if (t<=n) ins(,n,root[i-],root[i],h[t]);
else
{
root[i]=root[i-];
if (!st[t]) st[t]=i; else ed[t]=i;
}
}
}
int valfind(int x,int val)
{
for(int i=;i>=;i--)
if(deep[x]>=bin[i]&&mx[x][i]<=val)x=f[x][i];
return x;
}
int query(int l,int r,int x,int y,int val)
{
if (l==r) return l;
int mid=(l+r)>>;
int kk=sum[son[y][]]-sum[son[x][]],t;
//cout<<" "<<kk<<endl;
if (kk>=val) return query(l,mid,son[x][],son[y][],val);
else return query(mid+,r,son[x][],son[y][],val-kk);
}
void solve()
{
while (q--)
{
int x=read(),val=read(),k=read();
if (lastans!=-) x^=lastans,val^=lastans,k^=lastans;
int t=valfind(x,val);
int a=root[st[t]],b=root[ed[t]];
if (sum[b]-sum[a]<k) lastans=-;
else lastans=list[query(,n,a,b,sum[b]-sum[a]-k+)];
printf("%d\n",lastans);
} }
void prework()
{
bin[]=;
for (int i=; i<; i++) bin[i]=bin[i-]<<;
for (int i=; i<=*n; i++) fa[i]=i;
sort(list+,list++n);
for (int i=; i<=n; i++) h[i]=lower_bound(list+,list++n,h[i])-list;
for (int i=; i<=m; i++)
{
a[i].u=read(); a[i].v=read(); a[i].val=read();
}
build();
}
int main()
{
n=read(); m=read(); q=read();
for (int i=; i<=n; i++) list[i]=h[i]=read();
prework();
solve();
return ;
}
bzoj3551的更多相关文章
- bzoj3551 3545
我直接来讲在线好了 这是一个很巧妙的方法,把边作为一个点 做一遍最小生成树,当加如一条边时,我们把这条边两点x,y的并查集的根i,j的父亲都设为这条边代表的点k,由k向i,j连边 这样我们就构建出一棵 ...
- [您有新的未分配科技点][BZOJ3545&BZOJ3551]克鲁斯卡尔重构树
这次我们来搞一个很新奇的知识点:克鲁斯卡尔重构树.它也是一种图,是克鲁斯卡尔算法求最小生成树的升级版首先看下面一个问题:BZOJ3545 Peaks. 在Bytemountains有N座山峰,每座山峰 ...
- BZOJ3551 [ONTAK2010]Peaks加强版 kruskal 并查集 主席树 dfs序
欢迎访问~原文出处——博客园-zhouzhendong 去博客园看该题解 题目传送门 - BZOJ3551 题意概括 Description 在Bytemountains有N座山峰,每座山峰有他的高度 ...
- 【BZOJ3551】 [ONTAK2010]Peaks加强版
BZOJ3551 [ONTAK2010]Peaks加强版 Solution Kruscal重构树后发现可以对于小于的离散化然后倍增+主席树找到上一个的可行解. 然后就可以了. 如果数组开的不好,容易在 ...
- 【BZOJ3551】Peaks加强版(Kruskal重构树,主席树)
[BZOJ3551]Peaks加强版(Kruskal重构树,主席树) 题面 BZOJ Description 在Bytemountains有N座山峰,每座山峰有他的高度h_i.有些山峰之间有双向道路相 ...
- 【BZOJ3551】[ONTAK2010]Peaks加强版 最小生成树+DFS序+主席树
[BZOJ3545][ONTAK2010]Peaks Description 在Bytemountains有N座山峰,每座山峰有他的高度h_i.有些山峰之间有双向道路相连,共M条路径,每条路径有一个困 ...
- BZOJ3551 ONTAK2010Peaks加强版(kruskal重构树+dfs序+主席树)
kruskal重构树本质就是给并查集显式建树来替代可持久化并查集.将边按困难度从小到大排序后建出该树,按dfs序建主席树即可.查询时跳到深度最浅的满足在该重要度下已被合并的点,在子树内查询第k大. # ...
- 【bzoj3545/bzoj3551】[ONTAK2010]Peaks/加强版 Kruskal+树上倍增+Dfs序+主席树
bzoj3545 题目描述 在Bytemountains有N座山峰,每座山峰有他的高度h_i.有些山峰之间有双向道路相连,共M条路径,每条路径有一个困难值,这个值越大表示越难走,现在有Q组询问,每组询 ...
- [BZOJ3551][ONTAK2010]Peaks(加强版)(Kruskal重构树,主席树)
3551: [ONTAK2010]Peaks加强版 Time Limit: 20 Sec Memory Limit: 128 MBSubmit: 2438 Solved: 763[Submit][ ...
- [BZOJ3551]Peaks
[BZOJ3551]Peaks BZOJ luogu 建Kruskal重构树,点权为边权 按dfn序建出主席树 倍增找到能跳到的最浅的祖先 主席树查询一下 #include<bits/stdc+ ...
随机推荐
- 第13章 Swing程序组件----常用布局管理器
在Swing中,每个组件在容器中都有一个具体的位置和大小,而在容器中摆放各种组件时很难判断其具体位置和大小.布局管理器提供了Swing组件安排.展示在容器中的方法及基本的布局功能. Swing提供的常 ...
- 注意:MainActivity的oncreate方法里不要再inflate布局了(MainActivity里的点击事件无响应)
activity_main已经通过setContentView(R.layout.activity_main);设置给MainActivity, 不要再inflate出新布局,然后findviewby ...
- linux创建vg、lv
LVM磁盘管理 一.LVM简介... 1 二. LVM基本术语... 2 三. 安装LVM... 3 四. 创建和管理LVM... 4 2. 创建PV.. 6 3. 创建VG.. 7 4. 创建LV. ...
- android使用百度app分享,app统计出现的异常,FrontiaApplication类
想在app里加入百度分享和百度统计.查看了百度移动统计的文档后下载官网给的demo.参照demo给出的代码给自己的app加入代码.以上步骤比较简单.不细说,下面说下此过程需要注意的一些点,不然会引发异 ...
- 连接Oracle数据库的Hibernate配置文件
连接Oracle数据库的Hibernate配置文件连接Oracle的Hibernate配置文件有两种格式,一种是xml格式的,另一种是Java属性文件格式的.下面分别给出这两种格式配置文件的代码. 1 ...
- listView上拉刷新下拉加载
xlistview_header.xml <?xml version="1.0" encoding="utf-8"?> <RelativeLa ...
- H5 video跨域问题
事情是这样的,今天上午下班前,朋友发我一个js文件,说视频不播放,核心代码大概是这样子的: var player = document.createElement('video'); var sour ...
- jQuery实现瀑布流(pc、移动通用)
使用 jQuery 的 Masonry 插件来实现这种页面形式 1,分别下载 jQuery 与 Masonry ,然后把他们都加载到页面中使用. 加载代码: <script src=" ...
- ubuntu上安装vsftp-使用java进行匿名链接
检查环境: 1. 检查是否装过了ftp服务器 如果没有提示内容折,本机没有安装. root@hadoops:~# rpm -qa|grep vsftpdroot@hadoops:~# rpm -qa| ...
- 配置 Gitblit 进行 Git 代码管理
配置 Gitblit 进行 Git 代码管理 环境 CentOS 7 x64 IP: 10.6.0.2 首先需要安装jdk 安装步骤 就略过了 下载最新版本 gitblit wget http:/ ...