这题是裸的主席树,每个节点建一棵主席树,再加个lca就可以了。

  历尽艰辛,终于A掉了这一题,这般艰辛也显示出了打代码的不熟练。

  错误:1、lca倍增的时候i和j写反了,RE了5次,实在要吸取教训

     2、主席树插入操作的时候,如果插入到的那个点(叶节点)原来有值,而没有加上,导致了WA

  以下是历尽艰辛的代码,还很长。

 

#include <cstdio>
#include <cstdlib>
#include <algorithm>
#include <queue>
#include <map> using namespace std; const int maxn = ;
int n, m, w[maxn];
int head[maxn], label;
struct Edge
{
int v, next;
Edge (int v = , int next = ):
v(v), next(next) {}
}e[maxn*];
int depth[maxn], f[maxn][], root[maxn], temp[maxn], s_num;
queue <int> q;
map <int, int> id;
struct Tree
{
int sum[maxn*], ls[maxn*], rs[maxn*], cnt;
Tree ()
{
sum[] = , cnt = ;
}
void PushUp(int rt)
{
sum[rt] = sum[ls[rt]]+sum[rs[rt]];
}
void insert(int las_rt, int rt, int l, int r, int p, int d)
{
if (l == r)
{
sum[rt] = sum[las_rt]+d;
return ;
}
int mid = (l+r)>>;
if (p <= mid)
{
ls[rt] = ++cnt, rs[rt] = rs[las_rt];
insert(ls[las_rt], ls[rt], l, mid, p, d);
}
else
{
ls[rt] = ls[las_rt], rs[rt] = ++cnt;
insert(rs[las_rt], rs[rt], mid+, r, p, d);
}
PushUp(rt);
}
int query(int u_rt, int v_rt, int lca_rt, int lca_pos, int l, int r, int k)
{
if (l == r)
return l;
int mid = (l+r)>>;
int s_l = sum[ls[u_rt]]+sum[ls[v_rt]]-*sum[ls[lca_rt]]+(lca_pos >= l && lca_pos <= mid);
if (k <= s_l)
return query(ls[u_rt], ls[v_rt], ls[lca_rt], lca_pos, l, mid, k);
else
return query(rs[u_rt], rs[v_rt], rs[lca_rt], lca_pos, mid+, r, k-s_l);
}
}T; void ins(int u, int v)
{
e[++label] = Edge(v, head[u]);
head[u] = label;
} void in()
{
scanf("%d %d", &n, &m);
for (int i = ; i <= n; ++i)
scanf("%d", &w[i]);
for (int i = ; i <= n; ++i)
head[i] = -;
label = -;
for (int i = ; i < n; ++i)
{
int u, v;
scanf("%d %d", &u, &v);
ins(u, v), ins(v, u);
}
} void Build_lca()
{
for (int i = ; i <= ; ++i)
for (int j = ; j <= n; ++j)
f[j][i] = -;
q.push();
depth[] = ;
f[][] = -;
while (!q.empty())
{
int u = q.front();
for (int i = head[u]; i != -; i = e[i].next)
{
int v = e[i].v;
if (v == f[u][])
continue ;
f[v][] = u;
depth[v] = depth[u]+;
q.push(v);
}
q.pop();
}
for (int i = ; i <= ; ++i)
for (int j = ; j <= n; ++j)
{
if (f[j][i-] == -)
continue ;
f[j][i] = f[f[j][i-]][i-];
}
} void Hash_a()
{
for (int i = ; i <= n; ++i)
temp[i] = w[i];
sort(temp+, temp+n+);
s_num = ;
for (int i = ; i <= n; ++i)
if (temp[i] != temp[i-] || i == )
{
temp[++s_num] = temp[i];
id[temp[i]] = s_num;
}
} void dfs(int u)
{
root[u] = ++T.cnt;
T.insert(u == ? : root[f[u][]], root[u], , s_num, id[w[u]], );
for (int i = head[u]; i != -; i = e[i].next)
{
int v = e[i].v;
if (v == f[u][])
continue ;
dfs(v);
}
} void Build_tree()
{
Hash_a();
dfs();/*
for (int i = 1; i <= n; ++i)
printf("%d %d\n", i, T.sum[root[i]]);*/
} void prepare()
{
Build_lca();
Build_tree();
} int lca(int u, int v)
{
if (depth[u] < depth[v])
swap(u, v);
for (int i = ; i >= ; --i)
{
if (f[u][i] == -)
continue ;
if (depth[f[u][i]] >= depth[v])
{
u = f[u][i];
if (depth[u] == depth[v])
break ;
}
}
if (u == v)
return u;
for (int i = ; i >= ; --i)
{
if (f[u][i] == -)
continue ;
if (f[u][i] != f[v][i])
{
u = f[u][i];
v = f[v][i];
}
}
return f[u][];
} void work()
{
prepare();
while (m --)
{
int u, v, k;
scanf("%d %d %d", &u, &v, &k);
int t = lca(u, v);
int pos = T.query(root[u], root[v], root[t], id[w[t]], , s_num, k);
printf("%d\n", temp[pos]);
}
} int main()
{
in();
work();
return ;
}

  

SPOJ 10628. SPOJ COT Count on a tree 可持久化线段树的更多相关文章

  1. BZOJ - 2588 Spoj 10628. Count on a tree (可持久化线段树+LCA/树链剖分)

    题目链接 第一种方法,dfs序上建可持久化线段树,然后询问的时候把两点之间的所有树链扒出来做差. #include<bits/stdc++.h> using namespace std; ...

  2. SPOJ COT Count on a tree(树上主席树 + LCA 求点第k小)题解

    题意:n个点的树,每个点有权值,问你u~v路径第k小的点的权值是? 思路: 树上主席树就是每个点建一棵权值线段树,具体看JQ博客,LCA用倍增logn求出,具体原理看这里 树上主席树我每个点的存的是点 ...

  3. SPOJ 10628 COT - Count on a tree(在树上建立主席树)(LCA)

    COT - Count on a tree #tree You are given a tree with N nodes.The tree nodes are numbered from 1 to ...

  4. SPOJ - COT Count on a tree

    地址:http://www.spoj.com/problems/COT/en/ 题目: COT - Count on a tree #tree You are given a tree with N  ...

  5. 主席树 || 可持久化线段树 || LCA || BZOJ 2588: Spoj 10628. Count on a tree || Luogu P2633 Count on a tree

    题面: Count on a tree 题解: 主席树维护每个节点到根节点的权值出现次数,大体和主席树典型做法差不多,对于询问(X,Y),答案要计算ans(X)+ans(Y)-ans(LCA(X,Y) ...

  6. BZOJ 2588: Spoj 10628. Count on a tree-可持久化线段树+LCA(点权)(树上的操作) 无语(为什么我的LCA的板子不对)

    2588: Spoj 10628. Count on a tree Time Limit: 12 Sec  Memory Limit: 128 MBSubmit: 9280  Solved: 2421 ...

  7. 主席树[可持久化线段树](hdu 2665 Kth number、SP 10628 Count on a tree、ZOJ 2112 Dynamic Rankings、codeforces 813E Army Creation、codeforces960F:Pathwalks )

    在今天三黑(恶意评分刷上去的那种)两紫的智推中,突然出现了P3834 [模板]可持久化线段树 1(主席树)就突然有了不详的预感2333 果然...然后我gg了!被大佬虐了! hdu 2665 Kth ...

  8. 「SP10628 COT - Count on a tree」

    主席树的综合运用题. 前置芝士 可持久化线段树:其实就是主席树了. LCA:最近公共祖先,本题需要在\(\log_2N\)及以内的时间复杂度内解决这个问题. 具体做法 主席树维护每个点到根节点这一条链 ...

  9. SPOJ Meteors - 可持久化线段树 - 二分法

    Byteotian Interstellar Union (BIU) has recently discovered a new planet in a nearby galaxy. The plan ...

随机推荐

  1. 使用wifite破解路由器密码

    使用wifite破解路由器密码 发表于 2016-02-06   |   分类于 wifite  |   暂无评论  |   10次阅读 简介 wifite是一款自动化wep.wpa破解工具,不支持w ...

  2. 深入理解Spring系列之四:BeanDefinition装载前奏曲

    转载 https://mp.weixin.qq.com/s?__biz=MzI0NjUxNTY5Nw==&mid=2247483835&idx=1&sn=276911368d4 ...

  3. ProxySQL 监控和统计

    ProxySQL 监控和统计 很多有价值的统计数据在stats和monitor库中. admin@127.0.0.1 [(none)]>SHOW TABLES FROM stats; +---- ...

  4. WPF之换肤

    WPF之换肤 设计原理 WPF换肤的设计原理,利用资源字典为每种皮肤资源添加不同的样式,在后台切换皮肤资源文件. 截图 上图中,第一张图采用规则样式,第二张图采用不规则样式,截图的时候略有瑕疵. 资源 ...

  5. java.lang.IllegalArgumentException: Page directive: invalid value for import

    我的项目原来用的tomcat版本是apache-tomcat-7.0.53,后来为了安全原因将版本升至\apache-tomcat-7.0.57,发现有的jsp页面出现下面的异常: java.lang ...

  6. 2017百度春招<不等式排列>

    题目: 度度熊最近对全排列特别感兴趣,对于1到n的一个排列,度度熊发现可以在中间根据大小关系插入合适的大于和小于符号(即 '>' 和 '<' )使其成为一个合法的不等式数列.但是现在度度熊 ...

  7. /bin、/sbin、/usr/bin、/usr/sbin目录Linux执行文档的区别

    /bin./sbin./usr/bin./usr/sbin目录的区别   在linux下我们经常用到的四个应用程序的目录是/bin./sbin./usr/bin./usr/sbin .而四者存放的文件 ...

  8. docvalue and fielddata

    大部分字段类型默认被索引的(inverted index),可以被搜索search: 哪个文档有这个词sort&aggregations: look up the document and f ...

  9. Linux下的格式化字符串漏洞利用姿势

    linux最早的漏洞防护机制nx-stack刚刚出现后就有人想出了突破方法.那就是只有栈是不可执行,而除了栈以外的其他地方还是可以执行的,只要把返回地址执行别的地方就可以. 一.格式化字符串漏洞 格式 ...

  10. python中__dict__与dir()的区别

    在python中__dict__与dir()都可以返回一个对象的属性,区别在于: __dict__是对象的一个属性,而dir()是一个built-in的方法: __dict__返回一个对象的属性名和值 ...