题面传送门


复出的第一道题.. md就遇到坑了..

简单来说就是可持久化线段树+启发式合并啊..

感觉启发式合并好神奇好想学

每一次建边就暴力合并,每一个节点维护从根到它的权值线段树

按照题面的话最省空间的做法就是垃圾回收,但是实在是太慢了..

而且这题有坑,题面说的是多组数据其实只有一组 而且是$T>1$的一组..

然后看给了512MB就不需要垃圾回收,而且很多预处理都tm不用了呢!wqnmdsy

#include <cstdio>
#include <cstring>
#include <cstdlib>
#include <algorithm>
using namespace std;
const int Maxn = 80010;
const int lg = 18;
struct node {
int y, next;
}a[Maxn*2]; int first[Maxn], len;
void ins ( int x, int y ){
len ++;
a[len].y = y;
a[len].next = first[x]; first[x] = len;
}
int n, m, t;
int sum[Maxn*lg*lg], lc[Maxn*lg*lg], rc[Maxn*lg*lg], tot;
int rt[Maxn];
int na[Maxn], b[Maxn], bl;
int fa[Maxn][lg], dep[Maxn];
int faa[Maxn], gs[Maxn];
int ff ( int x ){
if ( faa[x] == x ) return x;
return faa[x] = ff (faa[x]);
}
void merge ( int &now, int fnow, int l, int r, int x ){
if ( !now ) now = ++tot;
sum[now] = sum[fnow]+1;
if ( l == r ) return;
int mid = ( l + r ) >> 1;
if ( x <= mid ) merge ( lc[now], lc[fnow], l, mid, x ), rc[now] = rc[fnow];
else merge ( rc[now], rc[fnow], mid+1, r, x ), lc[now] = lc[fnow];
}
void dfs1 ( int x, int f ){
rt[x] = 0;
merge ( rt[x], rt[fa[x][0]], 1, bl, na[x] );
for ( int i = 1; i <= 17; i ++ ) fa[x][i] = fa[fa[x][i-1]][i-1];
for ( int k = first[x]; k; k = a[k].next ){
int y = a[k].y;
if ( y == f ) continue;
fa[y][0] = x; dep[y] = dep[x]+1;
dfs1 ( y, x );
}
}
void change ( int &now, int l, int r, int x ){
if ( !now ) now = ++tot;
sum[now] ++;
if ( l == r ) return;
int mid = ( l + r ) >> 1;
if ( x <= mid ) change ( lc[now], l, mid, x );
else change ( rc[now], mid+1, r, x );
}
int getlca ( int x, int y ){
if ( dep[x] < dep[y] ) swap ( x, y );
for ( int i = 17; i >= 0; i -- ){
if ( dep[fa[x][i]] >= dep[y] ){
x = fa[x][i];
}
}
if ( x == y ) return x;
for ( int i = 17; i >= 0; i -- ){
if ( fa[x][i] != fa[y][i] ){
x = fa[x][i]; y = fa[y][i];
}
}
return fa[x][0];
}
int getrank ( int now1, int now2, int now3, int p, int l, int r, int k ){
if ( l == r ) return b[l];
int mid = ( l + r ) >> 1;
int ls = sum[lc[now1]]+sum[lc[now2]]-2*sum[lc[now3]];
if ( na[p] <= mid && na[p] >= l ) ls --;
if ( ls >= k ) return getrank ( lc[now1], lc[now2], lc[now3], p, l, mid, k );
else return getrank ( rc[now1], rc[now2], rc[now3], p, mid+1, r, k-ls );
}
void read ( int &x ){
char c = getchar ();
for ( ; c > '9' || c < '0'; c = getchar () );
x = 0;
for ( ; c <= '9' && c >= '0'; c = getchar () ) x = x*10+c-'0';
}
int main (){
int i, j, k, T;
read (T);
tot = 0;
while ( T -- ){
len = 0; memset ( first, 0, sizeof (first) );
read (n); read (m); read (t);
for ( i = 1; i <= n; i ++ ) rt[i] = 0, faa[i] = i, gs[i] = 1, dep[i] = 1;
for ( i = 1; i <= n; i ++ ){
read (na[i]);
b[i] = na[i];
}
sort ( b+1, b+n+1 );
bl = unique ( b+1, b+n+1 ) - (b+1);
for ( i = 1; i <= n; i ++ ){
na[i] = lower_bound ( b+1, b+bl+1, na[i] ) - b;
change ( rt[i], 1, bl, na[i] );
}
for ( i = 1; i <= m; i ++ ){
int x, y;
read (x); read (y);
int fx = ff (x), fy = ff (y);
if ( gs[fx] > gs[fy] ) swap ( x, y );
faa[fx] = fy; gs[fy] += gs[fx];
fa[x][0] = y; dep[x] = dep[y]+1;
dfs1 ( x, y );
ins ( x, y ); ins ( y, x );
}
char c;
int lastans = 0;
while ( t -- ){
c = getchar ();
for ( ; c > 'Z' || c < 'A'; c = getchar () );
if ( c == 'Q' ){
int x, y;
read (x); read (y); read (k);
x ^= lastans; y ^= lastans; k ^= lastans;
int lca = getlca ( x, y );
lastans = getrank ( rt[x], rt[y], rt[fa[lca][0]], lca, 1, bl, k );
printf ( "%d\n", lastans );
}
else {
int x, y;
read (x); read (y);
x ^= lastans; y ^= lastans;
int fx = ff (x), fy = ff (y);
if ( gs[fx] > gs[fy] ) swap ( x, y );
faa[fx] = fy; gs[fy] += gs[fx];
fa[x][0] = y; dep[x] = dep[y]+1;
dfs1 ( x, y );
ins ( x, y ); ins ( y, x );
}
}
break;
}
return 0;
}

  

bzoj3123: [Sdoi2013]森林的更多相关文章

  1. [bzoj3123][Sdoi2013]森林_主席树_启发式合并

    森林 bzoj-3123 Sdoi-2013 题目大意:给定一片共n个点的森林,T个操作,支持:连接两个不在一棵树上的两个点:查询一棵树上路径k小值. 注释:$1\le n,T \le 8\cdot ...

  2. [BZOJ3123][Sdoi2013]森林 主席树+启发式合并

    3123: [Sdoi2013]森林 Time Limit: 20 Sec  Memory Limit: 512 MB Description Input 第一行包含一个正整数testcase,表示当 ...

  3. BZOJ3123: [Sdoi2013]森林(启发式合并&主席树)

    3123: [Sdoi2013]森林 Time Limit: 20 Sec  Memory Limit: 512 MBSubmit: 4813  Solved: 1420[Submit][Status ...

  4. [bzoj3123][sdoi2013森林] (树上主席树+lca+并查集启发式合并+暴力重构森林)

    Description Input 第一行包含一个正整数testcase,表示当前测试数据的测试点编号.保证1≤testcase≤20. 第二行包含三个整数N,M,T,分别表示节点数.初始边数.操作数 ...

  5. 【主席树 启发式合并】bzoj3123: [Sdoi2013]森林

    小细节磕磕碰碰浪费了半个多小时的时间 Description Input 第一行包含一个正整数testcase,表示当前测试数据的测试点编号.保证1≤testcase≤20. 第二行包含三个整数N,M ...

  6. BZOJ3123[Sdoi2013]森林——主席树+LCA+启发式合并

    题目描述 输入 第一行包含一个正整数testcase,表示当前测试数据的测试点编号.保证1≤testcase≤20. 第二行包含三个整数N,M,T,分别表示节点数.初始边数.操作数.第三行包含N个非负 ...

  7. bzoj千题计划258:bzoj3123: [Sdoi2013]森林

    http://www.lydsy.com/JudgeOnline/problem.php?id=3123 启发式合并主席树 #include<cmath> #include<cstd ...

  8. [bzoj3123] [SDOI2013]森林 主席树+启发式合并+LCT

    Description Input 第一行包含一个正整数testcase,表示当前测试数据的测试点编号.保证1≤testcase≤20. 第二行包含三个整数N,M,T,分别表示节点数.初始边数.操作数 ...

  9. bzoj3123 [Sdoi2013]森林 树上主席树+启发式合并

    题目传送门 https://lydsy.com/JudgeOnline/problem.php?id=3123 题解 如果是静态的查询操作,那么就是直接树上主席树的板子. 但是我们现在有了一个连接两棵 ...

随机推荐

  1. iOS 25个性能优化/内存优化常用方法

    1. 用ARC管理内存 ARC(Automatic ReferenceCounting, 自动引用计数)和iOS5一起发布,它避免了最常见的也就是经常是由于我们忘记释放内存所造成的内存泄露.它自动为你 ...

  2. CI加载model的问题

    1.需求 CI在linux上无法加载model 2.原因 因为linux区分大小写,且model文件名首字符要大写. As said in the comments : Your model's fi ...

  3. instanceof运算符

    instanceof运算符:判断该对象是否是某一个类的实例. 语法格式:boolean b = 对象A instanceof 类B://判断A对象是否是B类的实例,如果是返回true. 若对象是类的实 ...

  4. How the problem solved about " Dealing with non-fast-forward errors"

    Recently ,I got confused When I use  git to push one of my project. The problem is below: And I Foun ...

  5. ng-repeat里创建的自定义指令

    在ng里,所有的指令在按照意愿正常工作之前的都需要编译一下,包含angularJS的自定义指令. ng模板里的所有指令都会在angularJS加载完毕之后编译一下,所以那些自定义指令和事件才能工作. ...

  6. 阿里云部署nodejs服务器(windows)

    花了大半个月做的网站终于要上线了,周围的同学们很多都在使用阿里云的服务器,我也入手了一台.考虑到自己不是很适应ubuntu的命令行界面,于是买了个windows的,上网搜了一下,似乎都是用linux来 ...

  7. xml编辑器

    cstring转cha型方法在mfc中用过可行 int CstringToch(CString str, char *ch) { assert(ch); memset(ch, 0, sizeof(ch ...

  8. css3 自定义字体的使用方法

    @font-face是CSS3中的一个模块,他主要是把自己定义的Web字体嵌入到你的网页中,随着@font-face模块的出现,我们在Web的开发中使用字体不怕只能使用Web安全字体,你们当中或许有许 ...

  9. CAD的输出成高清jpg图片

    打印名称选择JPG或者PNG 然后图纸尺寸选择大的 尺寸不够大就自己设置下 创建新图纸——设置下长宽——然后保存下名字,然后图纸尺寸选择你设置过的这个输出就好了 然后窗口下就好了

  10. 得静下心来看python了。

    陆陆续续看了些python爬虫的相关知识.今天找到了这篇博客不错就看这个,明天看到那个又很好又跑去看那个. 来来回回耽误不少时间.接下来不能再这样了,老老实实就看xlzd.me博客和Python实战: ...