SPOJ Count on a tree(主席树+LCA)
一、题目
COT - Count on a tree
You are given a tree with N nodes. The tree nodes are numbered from 1 to N. Each node has an integer weight.
We will ask you to perform the following operation:
- u v k : ask for the kth minimum weight on the path from node u to node v
Input
In the first line there are two integers N and M. (N, M <= 100000)
In the second line there are N integers. The ith integer denotes the weight of the ith node.
In the next N-1 lines, each line contains two integers u v, which describes an edge (u, v).
In the next M lines, each line contains three integers u v k, which means an operation asking for the kth minimum weight on the path from node u to node v.
Output
For each operation, print its result.
Example
Input: 8 5 105 2 9 3 8 5 7 7 1 2 1 3 1 4 3 5 3 6 3 7 4 82 5 12 5 22 5 32 5 47 8 2
Output: 2891057 题目链接:http://www.spoj.com/problems/COT/二、思路 原理其实和一维线性表序列的一样。只不过这是在树上操作。 在一维线性表序列中,扫描序列中每一个离散化后的数字,每一次新建的权值线段树都是基于前一次的。而在树中,对每一个子节点的数字,新建的线段树都是基于父节点的线段树的。
***************************未完待续……
***************************
三、源代码
#pragma GCC optimize(2) #pragma comment(linker, "/STACK:102400000, 102400000") #include<bits/stdc++.h> using namespace std; typedef long long LL; #define MAXN 111111 LL n, q, m; /*************树部分*******************/ typedef struct { int to, next; } Edge; Edge tree[MAXN * ]; int head[MAXN], ecnt; void add(int from, int to) { tree[ecnt].to = to; tree[ecnt].next = head[from]; head[from] = ecnt++; } /*************树部分*******************/ /*************LCA部分*******************/ ], depth[MAXN]; void dfs4lca(int r, int par, int d) { parent[r][] = par, depth[r] = d; ; i = tree[i].next) { int to = tree[i].to; if(to != par) dfs4lca(to, r, d + ); } } void init4lca() { dfs4lca(, -, ); ; k < ; ++k) { ; v <= n; ++v) { ) parent[v][k + ] = -; else parent[v][k + ] = parent[parent[v][k]][k]; } } } int lca(int a, int b) { if(depth[a] < depth[b]) swap(a, b); ; i < ; ++i) { ) a = parent[a][i]; } if(a == b) return a; ; i >= ; --i) { if(parent[a][i] != parent[b][i]) { a = parent[a][i]; b = parent[b][i]; } } ]; } /*************LCA部分*******************/ /*************主席树部分*******************/ struct { int lch, rch, cnt; } nodes[MAXN * ]; int root[MAXN], ncnt; , int r = m) { if(!nroot) { nroot = ++ncnt; nodes[nroot].cnt = nodes[proot].cnt + ; } if(l == r) return; ; if(val <= mid) { nodes[nroot].rch = nodes[proot].rch; update(nodes[nroot].lch, nodes[proot].lch, val, l, mid); } else { nodes[nroot].lch = nodes[proot].lch; update(nodes[nroot].rch, nodes[proot].rch, val, mid + , r); } } , int r = m) { if(l == r) return l; int cnt = nodes[nodes[uroot].lch].cnt + nodes[nodes[vroot].lch].cnt - nodes[nodes[lcaroot].lch].cnt - nodes[nodes[lcafroot].lch].cnt; ; if(k <= cnt) return query(nodes[uroot].lch, nodes[vroot].lch, nodes[lcaroot].lch, nodes[lcafroot].lch, k, l, mid); else , r); } /*************主席树部分*******************/ /*************离散化部分*******************/ LL weight[MAXN], buf[MAXN], mp[MAXN]; int nwt[MAXN]; void discrete() { memcpy(buf + , weight + , ]) * n); sort(buf + , buf + n + ); m = unique(buf + , buf + n + ) - buf - ; ; i <= n; ++i) { nwt[i] = lower_bound(buf + , buf + m + , weight[i]) - buf; mp[nwt[i]] = weight[i]; } } /*************离散化部分*******************/ template <class T> inline void read(T &x) { int t; bool flag = false; ')) ; if(t == '-') flag = true, t = getchar(); x = t - '; ') x = x * + t - '; if(flag) x = -x; } void init() { memset(head, -, sizeof(head)); ecnt = ; //nodes = 0, root = 0, ncnt = 0; } void dfs4tree(int r, int par) { ; i = tree[i].next) { int to = tree[i].to; if(to != par) { update(root[to], root[r], nwt[to]); dfs4tree(to, r); } } } int main() { #ifndef ONLINE_JUDGE freopen("input.txt", "r", stdin); #endif // ONLINE_JUDGE init(); LL a, b, u, v, k, wt, uvlca, ans; scanf("%lld%lld", &n, &q); ; i <= n; ++i) { read(weight[i]); } ; i < n; ++i) { read(a), read(b); add(a, b); add(b, a); } init4lca(); discrete(); add(, );//添加一条虚边。 dfs4tree(, -); while(q--) { read(u), read(v), read(k); uvlca = LL(lca(u, v)); ans = mp[query(root[u], root[v], root[uvlca], root[parent[uvlca][]], int(k))]; printf("%lld\n", ans); } ; }
SPOJ Count on a tree(主席树+LCA)的更多相关文章
- 【BZOJ2588】Spoj 10628. Count on a tree 主席树+LCA
[BZOJ2588]Spoj 10628. Count on a tree Description 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lasta ...
- BZOJ 2588: Spoj 10628. Count on a tree 主席树+lca
分析:树上第k小,然后我想说的是主席树并不局限于线性表 详细分析请看http://www.cnblogs.com/rausen/p/4006116.html,讲的很好, 然后因为这个熟悉了主席树,真是 ...
- spoj COT - Count on a tree(主席树 +lca,树上第K大)
您将获得一个包含N个节点的树.树节点的编号从1到Ñ.每个节点都有一个整数权重. 我们会要求您执行以下操作: uvk:询问从节点u到节点v的路径上的第k个最小权重 输入 在第一行中有两个整数Ñ和中号.( ...
- [bzoj2588][count on a tree] (主席树+lca)
Description 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lastans和v这两个节点间第K小的点权.其中lastans是上一个询问的答案,初始 ...
- 洛谷P2633/bzoj2588 Count on a tree (主席树)
洛谷P2633/bzoj2588 Count on a tree 题目描述 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lastans和v这两个节点间第K ...
- Bzoj 2588: Spoj 10628. Count on a tree 主席树,离散化,可持久,倍增LCA
题目:http://www.lydsy.com/JudgeOnline/problem.php?id=2588 2588: Spoj 10628. Count on a tree Time Limit ...
- spoj cot: Count on a tree 主席树
10628. Count on a tree Problem code: COT You are given a tree with N nodes.The tree nodes are number ...
- 【BZOJ-2588】Count on a tree 主席树 + 倍增
2588: Spoj 10628. Count on a tree Time Limit: 12 Sec Memory Limit: 128 MBSubmit: 3749 Solved: 873[ ...
- 洛谷P2633 Count on a tree(主席树上树)
题目描述 给定一棵N个节点的树,每个点有一个权值,对于M个询问(u,v,k),你需要回答u xor lastans和v这两个节点间第K小的点权.其中lastans是上一个询问的答案,初始为0,即第一个 ...
随机推荐
- POJ 2253 Frogger(dijkstra变形)
http://poj.org/problem?id=2253 题意: 有两只青蛙A和B,现在青蛙A要跳到青蛙B的石头上,中间有许多石头可以让青蛙A弹跳.给出所有石头的坐标点,求出在所有通路中青蛙需要跳 ...
- python 列表求和
def sum_list(items): sum_numbers = for x in items: sum_numbers += x return sum_numbers print(sum_lis ...
- Java回顾之反射
在这一篇文章里,我们关注反射及其相关话题. 反射可以帮助我们查看指定类型中的信息.创建类型的实例,调用类型的方法.我们平时使用框架,例如Spring.EJB.Hibernate等都大量的使用了反射技术 ...
- 伪多项式时间 Pseudo-polynomial time
2018-03-15 14:20:08 伪多项式时间:如果一个算法的传统时间复杂度是多项式时间的,而标准时间复杂度不是多项式时间的,则我们称这个算法是伪多项式时间的. 想要理解“伪多项式时间”,我们需 ...
- 安装MySQL后要做的事
安装MySQL后要修改的配置 [mysql] default-character-set=utf8 [mysqld] # 关闭域名反解 skip_name_resolve # 每表一个独立的表空间文件 ...
- 流程设计器jQuery + svg/vml(Demo5 - 撤消与重做)
上篇完成了画线,接下来是撤消与重做. 代码:GoFlow_05.zip 演示地址:Demo 微信演示公众号: 另:Silverlight版 Silverlight版Demo
- SVN的 安装
关于svn的安装,如图分别是服务端和客户端 下载地址:http://subversion.apache.org/packages.html 1.双击VisualSVN,点下一步,选择目录,就可以,很简 ...
- tensorflow中tensor的静态维度和动态维度
tf中使用张量(tensor)这种数据结构来表示所有的数据,可以把张量看成是一个具有n个维度的数组或列表,张量会在各个节点之间流动,参与计算. 张量具有静态维度和动态维度. 在图构建过程中定义的张量拥 ...
- alsa-lib、alsa-utils移植
/************************************************************************** * alsa-lib.alsa-utils移植 ...
- mysql单列索引和联合索引的使用
1,首先要确定优化的目标,在什么样的业务场景下,表的大小等等.如果表比较小的话,可能都不需要加索引. 2,哪些字段可以建索引,一般都where.order by 或者 group by 后面的字段. ...