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,即第一个 ...
随机推荐
- QT 正则表达式无效
背景:写了一个判断IP地址合法的正则表达式,并让它应用在输入框中 代码如下 QRegExp rx_ip("^((2[0-4]\\d|25[0-5]|[01]?\\d\\d?)\\.){3}( ...
- [OSG]OSG的相关扩展
参考:osg官网 http://www.osgchina.org/index.php?view=article&id=176 http://trac.openscenegraph.org/pr ...
- 后端利用Redis队列及哈希实现定时推送提醒的三个思路
周煦辰 2016年8月31日 本文介绍了一下本人在开发过程中遇到"定时推送提醒"的需求的时候所思考的三种解决方案. 明确问题 首先明确一下这个需求可能包含的几个"坑&qu ...
- POJ 1797 kruskal 算法
题目链接:http://poj.org/problem?id=1797 开始题意理解错.不说题意了. 并不想做这个题,主要是想测试kruskal 模板和花式并查集的正确性. 已AC: /* 最小生成树 ...
- ENUMSTXT.H中的指针数组
/************************************************************************ ...
- ESET Smart Security – 免费90天(sv)
Eset 活动很多,还有beta测试,几乎可以免费使用Eset.萨尔瓦多.免费活动,3个月免费ESS/EAV活动地址: 点此进入 填写表格,给你免费3个月的 ESS\EAV 许可证.官方网站: 点此 ...
- Python基础学习----参数和返回值
# 函数的参数和返回值 # 4种组合方式 # 1.无参无返 # def methodone(): # 2.无参有返 def methodtwo(): a=10 return a # 3.有参无返 # ...
- C++可调用对象与函数表
c++的可调用对象 有 函数 函数指针 lambda表达式 bind的对象 重载了函数调用运算符的类 如何调用? 函数调用 void afuncToCall() { cout << &qu ...
- python爬虫错误
错误描述 TypeError: list indices must be integers or slices, not str 错误缘由 取标签属性的时候, find_all()函数与find()函 ...
- springboot项目中文件的下载(解决中文乱码问题)
最近使用springboot项目,一直以来文件都以英文格式存储,这次使用的是xls文件下载,文件名为中文的,特此记录下中文文件名的下载以及springboot中下载路径报错问题. 正文 在使用spri ...