BZOJ 3626 [LNOI2014]LCA 树剖+(离线+线段树 // 在线+主席树)
BZOJ 4012 [HNOI2015]开店 的弱化版,离线了,而且没有边权(长度).
两种做法
1 树剖+离线+线段树
这道题求的是一个点zzz与[l,r][l,r][l,r]内所有点的lcalcalca的深度之和.可以发现,dep[lca(u,v)]dep[lca(u,v)]dep[lca(u,v)]就等于从uuu到根与从vvv到根的路径的交集路径的长度.那么只要把[l,r][l,r][l,r]所有点到根的路径标记了,然后用zzz点往根跑统计答案就行了.这样的话差分一下,离线就可以处理了.
时间复杂度O(nlog2n)O(nlog^2n)O(nlog2n)
CODE
#include <vector>
#include <queue>
#include <cstdio>
#include <cctype>
#include <cstring>
#include <algorithm>
using namespace std;
typedef long long LL;
inline void read(int &num) {
char ch; int flg=1; while(!isdigit(ch=getchar()))if(ch=='-')flg=-flg;
for(num=0; isdigit(ch); num=num*10+ch-'0',ch=getchar()); num*=flg;
}
const int MAXN = 50005;
const int mod = 201314;
int n, tot, Q, dfn[MAXN], seq[MAXN], tmr, fir[MAXN], cnt;
struct edge { int to, nxt; }e[MAXN];
inline void add(int u, int v) {
e[cnt] = (edge){ v, fir[u] }, fir[u] = cnt++;
}
int dep[MAXN], fa[MAXN], sz[MAXN], top[MAXN], son[MAXN];
void dfs(int u) {
dep[u] = dep[fa[u]] + (sz[u]=1);
for(int v, i = fir[u]; ~i; i = e[i].nxt) {
dfs(v=e[i].to), sz[u] += sz[v];
if(sz[v] > sz[son[u]]) son[u] = v;
}
}
void dfs2(int u, int tp) {
top[u] = tp; dfn[u] = ++tmr;
if(son[u]) dfs2(son[u], tp);
for(int v, i = fir[u]; ~i; i = e[i].nxt)
if((v=e[i].to) != son[u]) dfs2(v, v);
}
struct node {
int tim, x, id, flg;
}q[MAXN<<1];
inline bool cmp(const node &A, const node &B) { return A.tim < B.tim; }
int ans[MAXN];
int lz[MAXN<<2], sum[MAXN<<2];
inline void upd(int i) { sum[i] = sum[i<<1] + sum[i<<1|1]; }
inline void mt(int i, int l, int r, int mid) {
if(lz[i]) {
(lz[i<<1] += lz[i]) %= mod;
(lz[i<<1|1] += lz[i]) %= mod;
(sum[i<<1] += 1ll * lz[i] * (mid-l+1) % mod) %= mod;
(sum[i<<1|1] += 1ll * lz[i] * (r-mid) % mod) %= mod;
lz[i] = 0;
}
}
void insert(int i, int l, int r, int x, int y) {
if(l == x && r == y) {
(lz[i] += 1) %= mod;
(sum[i] += r-l+1) %= mod;
return;
}
int mid = (l + r) >> 1;
mt(i, l, r, mid);
if(y <= mid) insert(i<<1, l, mid, x, y);
else if(x > mid) insert(i<<1|1, mid+1, r, x, y);
else insert(i<<1, l, mid, x, mid), insert(i<<1|1, mid+1, r, mid+1, y);
upd(i);
}
int query(int i, int l, int r, int x, int y) {
if(l == x && r == y) return sum[i];
int mid = (l + r) >> 1;
mt(i, l, r, mid);
int res = 0;
if(y <= mid) res = query(i<<1, l, mid, x, y);
else if(x > mid) res = query(i<<1|1, mid+1, r, x, y);
else res = (query(i<<1, l, mid, x, mid) + query(i<<1|1, mid+1, r, mid+1, y)) % mod;
upd(i);
return res;
}
inline void Modify(int x) {
while(x) insert(1, 1, n, dfn[top[x]], dfn[x]), x = fa[top[x]];
}
inline int Query(int x) {
int res = 0;
while(x) (res += query(1, 1, n, dfn[top[x]], dfn[x])) %= mod, x = fa[top[x]];
return res;
}
int main() {
read(n); read(Q);
memset(fir, -1, sizeof fir);
for(int i = 2; i <= n; ++i)
read(fa[i]), ++fa[i], add(fa[i], i);
dfs(1); dfs2(1, 1);
for(int i = 1, x, y, z; i <= Q; ++i) {
read(x), read(y), read(z); ++x, ++y, ++z;
q[++tot] = (node){ x-1, z, i, -1 }; //差分
q[++tot] = (node){ y , z, i, 1 };
}
sort(q + 1, q + tot + 1, cmp);
int now = 0;
for(int i = 1; i <= tot; ++i) {
while(now < q[i].tim) Modify(++now);
(ans[q[i].id] += q[i].flg * Query(q[i].x)) %= mod;
}
for(int i = 1; i <= Q; ++i)
printf("%d\n", (ans[i]+mod)%mod);
}
2 树剖+在线+主席树
沿用法1的思路,直接在主席树上找rrr的线段树与l−1l-1l−1的线段树相减就行了.但是主席树不能下传标记,那么就把标记永久化.查询的时候在主席树上从根往下边走边统计.
时间复杂度同样是O(nlog2n)O(nlog^2n)O(nlog2n),常数巨大…
CODE
#include <cmath>
#include <cstdio>
#include <cstring>
#include <algorithm>
using namespace std;
typedef long long LL;
template<typename T>inline void read(T &num) {
char ch; int flg = 1;
while((ch=getchar())<'0'||ch>'9')if(ch=='-')flg=-flg;
for(num=0;ch>='0'&&ch<='9';num=num*10+ch-'0',ch=getchar());
num*=flg;
}
const int MAXN = 50005;
const int mod = 201314;
const int MAXNN = MAXN*100;
int n, q, A, fir[MAXN], cnt;
struct edge { int to, nxt; }e[MAXN];
inline void add(int u, int v) {
e[cnt] = (edge){ v, fir[u] }, fir[u] = cnt++;
}
int son[MAXN], sz[MAXN], top[MAXN], tmr, dfn[MAXN], fa[MAXN];
void dfs(int u) {
sz[u] = 1;
for(int i = fir[u], v; ~i; i = e[i].nxt) {
dfs(v=e[i].to), sz[u] += sz[v];
if(sz[v] > sz[son[u]]) son[u] = v;
}
}
void dfs2(int u, int tp) {
top[u] = tp; dfn[u] = ++tmr;
if(son[u]) dfs2(son[u], tp);
for(int i = fir[u], v; ~i; i = e[i].nxt)
if((v=e[i].to) != fa[u] && v != son[u])
dfs2(v, v);
}
int ch[MAXNN][2], tim[MAXNN], tot, rt[MAXN], sum[MAXNN];
inline void Newnode(int i, int p) { ch[i][0] = ch[p][0], ch[i][1] = ch[p][1], sum[i] = sum[p], tim[i] = tim[p]; }
void modify(int &i, int l, int r, int L, int R) {
Newnode(++tot, i);
if(L == l && r == R) { (++tim[i = tot]) %= mod; return; }
(sum[i = tot] += R-L+1) %= mod;
int mid = (l + r) >> 1;
if(R <= mid) modify(ch[i][0], l, mid, L, R);
else if(L > mid) modify(ch[i][1], mid+1, r, L, R);
else modify(ch[i][0], l, mid, L, mid), modify(ch[i][1], mid+1, r, mid+1, R);
}
inline void Modify(int &r, int x) { while(x) modify(r, 1, n, dfn[top[x]], dfn[x]), x = fa[top[x]]; }
int query(int i, int l, int r, int L, int R) {
if(!i) return 0;
int res = 1ll * (R-L+1) * tim[i] % mod;
if(L == l && r == R) return (res + sum[i]) % mod;
int mid = (l + r) >> 1;
if(R <= mid) return (res + query(ch[i][0], l, mid, L, R)) % mod;
else if(L > mid) return (res + query(ch[i][1], mid+1, r, L, R)) % mod;
return (res + query(ch[i][0], l, mid, L, mid) + query(ch[i][1], mid+1, r, mid+1, R)) % mod;
}
int Query(int r, int x) { int res = 0; while(x) (res += query(r, 1, n, dfn[top[x]], dfn[x])) %= mod, x = fa[top[x]]; return res; }
int main () {
read(n), read(q);
memset(fir, -1, sizeof fir);
for(int i = 2; i <= n; ++i)
read(fa[i]), ++fa[i], add(fa[i], i);
dfs(1), dfs2(1, 1);
for(int i = 1; i <= n; ++i) Modify(rt[i]=rt[i-1], i);
int L, R, x;
while(q--) {
read(L), read(R), read(x); ++L, ++R, ++x;
printf("%d\n", (Query(rt[R], x) - Query(rt[L-1], x) + mod) % mod);
}
}
BZOJ 3626 [LNOI2014]LCA 树剖+(离线+线段树 // 在线+主席树)的更多相关文章
- bzoj 3626 [LNOI2014]LCA(离线处理+树链剖分,线段树)
3626: [LNOI2014]LCA Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 1272 Solved: 451[Submit][Status ...
- BZOJ 3626: [LNOI2014]LCA [树链剖分 离线|主席树]
3626: [LNOI2014]LCA Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 2050 Solved: 817[Submit][Status ...
- BZOJ 3626: [LNOI2014]LCA( 树链剖分 + 离线 )
说多了都是泪啊...调了这么久.. 离线可以搞 , 树链剖分就OK了... -------------------------------------------------------------- ...
- bzoj 3626: [LNOI2014]LCA 离线+树链剖分
3626: [LNOI2014]LCA Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 426 Solved: 124[Submit][Status] ...
- BZOJ 3626 [LNOI2014]LCA:树剖 + 差分 + 离线【将深度转化成点权之和】
题目链接:http://www.lydsy.com/JudgeOnline/problem.php?id=3626 题意: 给出一个n个节点的有根树(编号为0到n-1,根节点为0,n <= 50 ...
- [BZOJ 3626] [LNOI2014] LCA 【树链剖分 + 离线 + 差分询问】
题目链接: BZOJ - 3626 题目分析 考虑这样的等价问题,如果我们把一个点 x 到 Root 的路径上每个点的权值赋为 1 ,其余点的权值为 0,那么从 LCA(x, y) 的 Depth 就 ...
- BZOJ 3626 [LNOI2014]LCA ——树链剖分
思路转化很巧妙. 首先把询问做差分. 然后发现加入一个点就把路径上的点都+1,询问的时候直接询问到根的路径和. 这样和原问题是等价的,然后树链剖分+线段树就可以做了. #include <map ...
- HDU 6162 Ch’s gift (树剖 + 离线线段树)
Ch’s gift Time Limit: 6000/3000 MS (Java/Others) Memory Limit: 65536/65536 K (Java/Others)Total S ...
- P4211[BZOJ 3626] [LNOI2014]LCA
题目描述 给出一个n个节点的有根树(编号为0到n-1,根节点为0).一个点的深度定义为这个节点到根的距离+1. 设dep[i]表示点i的深度,LCA(i,j)表示i与j的最近公共祖先. 有q次询问,每 ...
随机推荐
- win10下访问vm虚拟机Linux服务
一.环境 win10操作系统 centos6.5 VMware® Workstation 14 Pro虚拟机 二.互相访问设置步骤 1.查看如下图所示 2.记住上面的IP和网关,进行如下图操作 点击应 ...
- 【计算机网络】-网络层-Internet的网络层
[计算机网络]-网络层-Internet的网络层 Internet是一组相互连接的网络或者自治系统的集合 Internet 1.存在几个主要骨干网络,骨干网络是由高带宽的线路和快速路由器构成 2.这些 ...
- k8s遇坑:The connection to the server k8s-api.virtual.local:6443 was refused - did you specify the right host or port?
k8s坑The connection to the server localhost:8080 was refused - did you specify the right host or port ...
- 牛客 109B 好位置 (字符串水题)
大意: 给定字符串$s1,s2$, 对于$s1$中所有与$s2$相等的子序列$t$, $t$在$s1$中的下标定义为好位置. 求$s1$是否所有位置都是好位置. 显然$s1$的前缀要与$s2$相等, ...
- 使用SQL语句查询Elasticsearch索引数据
Elasticsearch 的官方查询语言是 Query DSL,存在毕竟有存在的道理,存在即合理.SQL 作为一个数据库查询语言,它语法简洁,书写方便而且大部分服务端程序员都清楚了解和熟知它的写法. ...
- dev chart使用
public class LineChartHelp { #region 折线图 /// <summary> /// 创建折线图 /// </summary> public v ...
- Qt 中使用Java代码获取安卓设备的MAC地址(安卓9.0)
public String GetDeviceMAC() { String strMacAddr = null; try { // 获得IpD地址 InetAddress ip = getLocalI ...
- Linq操作之Except,Distinct,Left Join 【转】
最近项目中用到了Linq中Except,Distinct,Left Join这几个运算,这篇简单的记录一下这几种情形. Except 基础类型使用Linq的运算很简单,下面用来计算两个集合的 ...
- 服务端相关知识学习(二)之Zookeeper可以干什么
Zookeeper主要可以干哪些事情 配置管理,名字服务,提供分布式同步以及集群管理.那这些服务又到底是什么呢?我们为什么需要这样的服务?我们又为什么要使用Zookeeper来实现呢,使用Zookee ...
- 关于登陆界面,页面没有刷新完毕,点击登陆跳转到一个接口的bug
现象 输入完密码点击登陆就跳转到了如下的页面 分析原因: 第一:查看html页面 页面中的html 登陆用的是form表单 表单中还写了属性 action 即允许跳到某一个接口,这里是没 ...