小细节磕磕碰碰浪费了半个多小时的时间

Description

Input

第一行包含一个正整数testcase,表示当前测试数据的测试点编号。保证1≤testcase≤20。 
第二行包含三个整数N,M,T,分别表示节点数、初始边数、操作数。第三行包含N个非负整数表示 N个节点上的权值。 
 接下来 M行,每行包含两个整数x和 y,表示初始的时候,点x和点y 之间有一条无向边, 接下来 T行,每行描述一个操作,格式为“Q x y k”或者“L x y ”,其含义见题目描述部分。

Output

对于每一个第一类操作,输出一个非负整数表示答案。

HINT

对于第一个操作 Q 8 7 3,此时 lastans=0,所以真实操作为Q 8^0 7^0 3^0,也即Q 8 7 3。点8到点7的路径上一共有5个点,其权值为4 1 1 2 4。这些权值中,第三小的为 2,输出 2,lastans变为2。对于第二个操作 Q 3 5 1 ,此时lastans=2,所以真实操作为Q 3^2 5^2 1^2 ,也即Q 1 7 3。点1到点7的路径上一共有4个点,其权值为 1 1 2 4 。这些权值中,第三小的为2,输出2,lastans变为 2。之后的操作类似。


题目分析

这个题意已经是非常裸的数据结构题了。我们需要实现的有:只有加边的动态LCA、主席树启发式合并。

本来还想写个内存回收的,但是写完一来发现对效率好像不是很自信;二来看了看觉得空间也还过得去,就没再写回收了。

没有理解为什么森林初始不是全为点的。可能是为了区分我这样的大常数?

话说20s的题交上去刷新一下就返回TLE是什么机制……

 #include<bits/stdc++.h>
const int maxn = ;
const int maxm = ;
const int maxNode = ; struct segNode
{
int l,r,val;
}a[maxNode];
struct Pool
{
int tot;
int newNode()
{
++tot, a[tot].l = a[tot].r = a[tot].val = ;
return tot;
}
}node;
char opt[];
int T,n,m,q,lastans;
int rt[maxn],c[maxn],tmp[maxn],fat[maxn],size[maxn];
int edgeTot,head[maxn],nxt[maxm],edges[maxm];
struct treeStruction
{
int fa[maxn][],dep[maxn];
void init()
{
memset(fa, , sizeof fa);
for (int i=; i<=n; i++) dep[i] = ;
}
int lca(int u, int v)
{
if (dep[u] < dep[v]) std::swap(u, v);
for (int i=; i>=; i--)
if (dep[fa[u][i]] >= dep[v]) u = fa[u][i];
if (u==v) return u;
for (int i=; i>=; i--)
if (fa[u][i]!=fa[v][i])
u = fa[u][i], v = fa[v][i];
return fa[u][];
}
}tre; int read()
{
char ch = getchar();
int num = , fl = ;
for (; !isdigit(ch); ch=getchar())
if (ch=='-') fl = -;
for (; isdigit(ch); ch=getchar())
num = (num<<)+(num<<)+ch-;
return num*fl;
}
int find(int x)
{
while (x!=fat[x]) x = fat[x];
return x;
}
void update(int &rt, int pre, int L, int R, int c)
{
rt = node.newNode();
a[rt] = a[pre], ++a[rt].val;
if (L==R) return;
int mid = (L+R)>>;
if (c <= mid) update(a[rt].l, a[pre].l, L, mid, c);
else update(a[rt].r, a[pre].r, mid+, R, c);
}
int query(int u, int v, int lca, int fa, int L, int R, int k)
{
if (L==R) return L;
int mid = (L+R)>>, val = a[a[u].l].val+a[a[v].l].val-a[a[lca].l].val-a[a[fa].l].val;
if (k <= val) return query(a[u].l, a[v].l, a[lca].l, a[fa].l, L, mid, k);
return query(a[u].r, a[v].r, a[lca].r, a[fa].r, mid+, R, k-val);
}
void connect(int x, int fat)
{
tre.dep[x] = tre.dep[fat]+, tre.fa[x][] = fat;
update(rt[x], rt[fat], , tmp[], c[x]);
for (int i=; i<=; i++)
tre.fa[x][i] = tre.fa[tre.fa[x][i-]][i-];
for (int i=head[x]; i!=-; i=nxt[i])
if (fat!=edges[i]) connect(edges[i], x);
}
void addedge(int u, int v)
{
int fu = find(u), fv = find(v);
if (size[fu] > size[fv]) std::swap(fu, fv), std::swap(u, v);
fat[fu] = fv, size[fv] += size[fu], connect(u, v);
edges[++edgeTot] = v, nxt[edgeTot] = head[u], head[u] = edgeTot;
edges[++edgeTot] = u, nxt[edgeTot] = head[v], head[v] = edgeTot;
}
void discretization()
{
std::sort(tmp+, tmp+n+);
tmp[] = std::unique(tmp+, tmp+n+)-tmp-;
for (int i=; i<=n; i++)
{
c[i] = std::lower_bound(tmp+, tmp+tmp[]+, c[i])-tmp;
update(rt[i], , , tmp[], c[i]);
}
}
int main()
{
T = read();
memset(rt, , sizeof rt);
memset(head, -, sizeof head);
n = read(), m = read(), q = read();
tre.init();
node.tot = lastans = edgeTot = ;
for (int i=; i<=n; i++)
tmp[i] = c[i] = read(), fat[i] = i, size[i] = ;
discretization();
for (int i=; i<=m; i++) addedge(read(), read());
for (int i=; i<=q; i++)
{
scanf("%s",opt);
if (opt[]=='Q'){
int u = read()^lastans, v = read()^lastans;
int k = read()^lastans, lca = tre.lca(u, v);
lastans = tmp[query(rt[u], rt[v], rt[lca], rt[tre.fa[lca][]], , tmp[], k)];
printf("%d\n",lastans);
}else addedge(read()^lastans, read()^lastans);
}
return ;
}

END

【主席树 启发式合并】bzoj3123: [Sdoi2013]森林的更多相关文章

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

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

  2. 【bzoj3123】[Sdoi2013]森林 倍增LCA+主席树+启发式合并

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

  3. P3302 [SDOI2013]森林(主席树+启发式合并)

    P3302 [SDOI2013]森林 主席树+启发式合并 (我以前的主席树板子是错的.......坑了我老久TAT) 第k小问题显然是主席树. 我们对每个点维护一棵包含其子树所有节点的主席树 询问(x ...

  4. 【BZOJ-3123】森林 主席树 + 启发式合并

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

  5. BZOJ_3123_[Sdoi2013]森林_主席树+启发式合并

    BZOJ_3123_[Sdoi2013]森林_主席树+启发式合并 Description Input 第一行包含一个正整数testcase,表示当前测试数据的测试点编号.保证1≤testcase≤20 ...

  6. Bzoj 3123: [Sdoi2013]森林(主席树+启发式合并)

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

  7. Bzoj2534:后缀自动机 主席树启发式合并

    国际惯例的题面:考虑我们求解出字符串uvu第一个u的右端点为i,第二个u的右端点为j,我们需要满足什么性质?显然j>i+L,因为我们选择的串不能是空串.另外考虑i和j的最长公共前缀(也就是说其p ...

  8. 【主席树启发式合并】【P3302】[SDOI2013]森林

    Description 给定一个 \(n\) 个节点的森林,有 \(Q\) 次操作,每次要么将森林中某两点联通,保证操作后还是个森林,要么查询两点间权值第 \(k\) 小,保证两点联通.强制在线. L ...

  9. Bzoj 3673: 可持久化并查集 by zky(主席树+启发式合并)

    3673: 可持久化并查集 by zky Time Limit: 5 Sec Memory Limit: 128 MB Description n个集合 m个操作 操作: 1 a b 合并a,b所在集 ...

随机推荐

  1. Spring JdbcTemplate详解及项目中的运用

    1.Spring对不同的持久化支持: Spring为各种支持的持久化技术,都提供了简单操作的模板和回调 ORM持久化技术 模板类 JDBC org.springframework.jdbc.core. ...

  2. css Masks

    css Masks:添加蒙板: 测试在微信端可以支持了.谷歌浏览器支持.safari应该也是支持的. 效果:http://runjs.cn/code/xrrgmgmk 但是谷歌可以支持这样子的:htt ...

  3. CC38:第k个数

    题目 有一些数的素因子只有3.5.7,请设计一个算法,找出其中的第k个数. 给定一个数int k,请返回第k个数.保证k小于等于100. 测试样例: 3 返回:7 解法 主要就是在于isPrime这个 ...

  4. sublime下Docblocker插件自定义配置

    1.下载安装Docblocker插件 栏目preferences-->Package Control 输入 Docblocker 搜索插件,单击安装 简单使用: 在函数头部输入: /**   按 ...

  5. Java通过图片url地址获取图片base64位字符串的两种方式

    工作中遇到通过图片的url获取图片base64位的需求.一开始是用网上的方法,通过工具类Toolkit,虽然实现的代码比较简短,不过偶尔会遇到图片转成base64位不正确的情况,至今不知道为啥. 之后 ...

  6. eShopOnContainers(一)

    微软微服务架构eShopOnContainers(一) 为了推广.Net Core,微软为我们提供了一个开源Demo-eShopOnContainers,这是一个使用Net Core框架开发的,跨平台 ...

  7. 066 Plus One

    给定一个非负整数组成的非空数组,给整数加一.可以假设整数不包含任何前导零,除了数字0本身.最高位数字存放在列表的首位.详见:https://leetcode.com/problems/plus-one ...

  8. dynomite:高可用多数据中心同步

    https://github.com/Netflix/dynomite Dynomite, inspired by Dynamo whitepaper, is a thin, distributed ...

  9. 一个很好用的侧滑框架ICSDrawerController实现的 QQ 侧滑及换肤功能

    使用ICSDrawerController 实现侧滑功能 在ICSDrawerController 第三方上做了修改实现,QQ 点击头像打开关抽屉头像渐变的效果 - (void)hiddenHeadV ...

  10. 代码管理_Git中获取两个标签之间的变更代码行数

    操作步骤: 1.先进入git的服务器,定位到 repositories 目录 2.再定位到具体的项目目录(登录bitbucket查看项目属性可以看到项目的存放目录),如下图: 3.浏览项目的标签,命令 ...