COT2 - Count on a tree II

中文题意

离线询问一颗树上路径(u,v)中经过所有点的权值的种类数。

题解

树上莫队。即在树的欧拉序列上进行莫队。同一个点加第一次时增加,第二次时减去增加的影响。

错误记录

用了tarjan LCA,并偷懒将询问和莫队都用一个node表示。意味着每一个询问都要开两个node,写莫队排序sort的时候注意结构体数组长度!莫队排序后两个相同的node不一定相邻,因为可能出现node[i].l/size和node[j].l/size相等,但是node[i].l和node[j].l并不相等的情况。卡了半天。

题目信息不全,权值据说是1e9,别忘了离散化

#include <cstdio>
#include <cstring>
#include <algorithm>
#include <cstdlib>
#include <iostream>
#include <cmath> void swap(int &a, int &b){int tmp = a;a = b, b = tmp;}
int max(int a, int b){return a > b ? a : b;}
int min(int a, int b){return a < b ? a : b;}
void read(int &x)
{
x = 0;char ch = getchar(), c = ch;
while(ch < '0' || ch > '9') c = ch, ch = getchar();
while(ch <= '9' && ch >= '0') x = x * 10 + ch - '0', ch = getchar();
if(c == '-') x = -x;
} const int INF = 0x3f3f3f3f;
const int MAXN = 1000000;
const int MAXM = 1000000; struct Edge
{
int u, v, nxt;
Edge(int _u, int _v, int _nxt){u = _u, v = _v, nxt = _nxt;}
Edge(){}
}edge[MAXN << 1];
int head[MAXN], cnt;
void insert(int a, int b)
{
edge[++ cnt] = Edge(a, b, head[a]), head[a] = cnt;
edge[++ cnt] = Edge(b, a, head[b]), head[b] = cnt;
} int tong[MAXN], val[MAXN], num[MAXN], id[MAXN];
int fa[MAXN], seq[MAXN], st[MAXN], et[MAXN], t, sum;
int n, m; void dfs(int x)
{
seq[++ t] = x;
st[x] = t;
for(int pos = head[x];pos;pos = edge[pos].nxt)
{
int v = edge[pos].v;
if(v == fa[x]) continue;
fa[v] = x;
dfs(v);
}
seq[++ t] = x;
et[x] = t;
} int cmp(int a, int b)
{
return val[a] < val[b];
} struct Node
{
int l, r, id, lca, nxt, u, v, need_lca;
Node(int _u, int _v, int _id, int _nxt){u = _u, v = _v, id = _id, nxt = _nxt;}
Node(){}
void init()
{
if(lca == u)
l = st[lca], r = st[v], need_lca = 0;
else if(lca == v)
l = st[lca], r = st[u], need_lca = 0;
else
{
if(st[u] < st[v]) l = et[u], r = st[v];
else l = et[v], r = st[u];
need_lca = 1;
}
return ;
}
}node[MAXM];
int head_node[MAXM], cnt_node = 1; void insert_node(int u, int v, int id)
{
node[++ cnt_node] = Node(u, v, id, head_node[u]), head_node[u] = cnt_node;
node[++ cnt_node] = Node(v, u, id, head_node[v]), head_node[v] = cnt_node;
} int ans[MAXM]; int find(int x)
{
return fa[x] == x ? x : fa[x] = find(fa[x]);
} int vis[MAXN]; void dfs_tarjan(int x)
{
vis[x] = 1;
for(int pos = head[x];pos;pos = edge[pos].nxt)
{
int v = edge[pos].v;
if(vis[v]) continue;
dfs_tarjan(v);
int f1 = find(x), f2 = find(v);
fa[f2] = f1;
}
for(int pos = head_node[x];pos;pos = node[pos].nxt)
{
int v = node[pos].v;
if(vis[v])
node[pos].lca = node[pos ^ 1].lca = find(v);
}
} void tarjan_lca()
{
for(int i = 1;i <= n;++ i) fa[i] = i;
dfs_tarjan(1);
} int size; int cmp2(Node& a, Node& b)
{
return a.l/size == b.l/size ? a.r < b.r : a.l/size < b.l/size;
} int w[MAXN]; void add(int x)
{
if(w[x] == 0) tong[val[x]] += 1;
else if(w[x] == 1) tong[val[x]] -= 1; if(tong[val[x]] == 1 && w[x] == 0) ++ sum;
if(tong[val[x]] == 0 && w[x] == 1) -- sum; ++ w[x];
} void del(int x)
{
if(w[x] == 1) tong[val[x]] -= 1;
else if(w[x] == 2) tong[val[x]] += 1; if(tong[val[x]] == 1 && w[x] == 2) ++ sum;
if(tong[val[x]] == 0 && w[x] == 1) -- sum; -- w[x];
} bool is_cal[MAXM << 1]; int main()
{
read(n), read(m);
for(int i = 1;i <= n;++ i) read(val[i]), id[i] = i;
for(int i = 1;i < n;++ i)
{
int tmp1, tmp2;
read(tmp1), read(tmp2);
insert(tmp1, tmp2);
}
std::sort(id + 1, id + 1 + n, cmp); for(int i = 1, j = 1;i <= n;)
{
num[j] = val[id[i]];
while(val[id[i]] == num[j]) val[id[i]] = j, ++ i;
++ j;
} dfs(1); for(int i = 1;i <= m;++ i)
{
int tmp1, tmp2;
read(tmp1), read(tmp2);
insert_node(tmp1, tmp2, i);
} tarjan_lca(); for(int i = 2;i <= cnt_node;++ i)
node[i].init(); size = sqrt(n);
if(size == 0) size = 1;
std::sort(node + 2, node + 1 + cnt_node, cmp2); int l = 1, r = 1;
add(seq[1]);
for(int i = 2;i <= cnt_node;++ i)
{
if(is_cal[node[i].id]) continue;
is_cal[node[i].id] = 1;
while(l < node[i].l) del(seq[l]), ++ l;
while(l > node[i].l) -- l, add(seq[l]);
while(r < node[i].r) ++ r, add(seq[r]);
while(r > node[i].r) del(seq[r]), -- r; if(node[i].need_lca) add(node[i].lca);
ans[node[i].id] = sum;
if(node[i].need_lca) del(node[i].lca);
}
for(int i = 1;i <= m;++ i)
printf("%d\n", ans[i]); return 0;
}

SPOJ10707 COT2-Count on a tree II的更多相关文章

  1. SPOJ10707 COT2 - Count on a tree II 【树上莫队】

    题目分析: 考虑欧拉序,这里的欧拉序与ETT欧拉序的定义相同而与倍增LCA不同.然后不妨对于询问$u$与$v$让$dfsin[u] \leq dfsin[v]$,这样对于u和v不在一条路径上,它们可以 ...

  2. 【SPOJ10707】 COT2 Count on a tree II

    SPOJ10707 COT2 Count on a tree II Solution 我会强制在线版本! Solution戳这里 代码实现 #include<stdio.h> #inclu ...

  3. spoj COT2 - Count on a tree II

    COT2 - Count on a tree II http://www.spoj.com/problems/COT2/ #tree You are given a tree with N nodes ...

  4. SPOJ COT2 - Count on a tree II(LCA+离散化+树上莫队)

    COT2 - Count on a tree II #tree You are given a tree with N nodes. The tree nodes are numbered from  ...

  5. COT2 - Count on a tree II(树上莫队)

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

  6. 「SPOJ10707」Count on a tree II

    「SPOJ10707」Count on a tree II 传送门 树上莫队板子题. 锻炼基础,没什么好说的. 参考代码: #include <algorithm> #include &l ...

  7. 【SPOJ10707】COT2 - Count on a tree II

    题目大意:给定一棵 N 个节点的无根树,每个节点有一个颜色.现有 M 个询问,每次询问一条树链上的不同颜色数. 题解:学会了树上莫队. 树上莫队是将节点按照欧拉序进行排序,将树上问题转化成序列上的问题 ...

  8. SPOJ COT2 Count on a tree II(树上莫队)

    题目链接:http://www.spoj.com/problems/COT2/ You are given a tree with N nodes.The tree nodes are numbere ...

  9. SPOJ COT2 Count on a tree II (树上莫队)

    题目链接:http://www.spoj.com/problems/COT2/ 参考博客:http://www.cnblogs.com/xcw0754/p/4763804.html上面这个人推导部分写 ...

  10. [SP10707]COT2 - Count on a tree II

    题目大意:有一棵$n$个节点的树,第$i$个点有一个颜色$C_i$,$m$组询问,每次问$x->y$的路径上有多少种颜色 题解:树上莫队,把树按欧拉序展开成一条链,令第$i$个节点第一次出现在序 ...

随机推荐

  1. sqoop2安装总结

    sqoop2安装 1. 下载解压缩 此次安装版本为1.99.6 # Decompress Sqoop distribution tarball tar -xvf sqoop-<version&g ...

  2. 记录一次失败的向git提交代码,和解决的方法。(首次创建仓库)

    背景: 向git push代码(创建一个新的仓库) 做法: 在github创建一个新的仓库------>本地新建文件夹------->依次执行了下面的命令 git init git clo ...

  3. 第36讲 谈谈MySQL支持的事务隔离级别,以及悲观锁和乐观锁的原理和应用场景

    在日常开发中,尤其是业务开发,少不了利用 Java 对数据库进行基本的增删改查等数据操作,这也是 Java 工程师的必备技能之一.做好数据操作,不仅仅需要对 Java 语言相关框架的掌握,更需要对各种 ...

  4. <router-link :to="...">

    一.<router-link :to="..."> to里的值可以是一个字符串路径,或者一个描述地址的对象.例如: // 字符串<router-link to=& ...

  5. JAVA API about HTTP 2

    import java.io.IOException; import java.nio.charset.Charset; import java.security.KeyManagementExcep ...

  6. bzoj1568 Blue Mary

    题意:P:加入一条一次函数.Q:询问x位置的最大函数值. 标程: #include<bits/stdc++.h> using namespace std; ; int q,x,n; dou ...

  7. js实现F5键刷新后菜单保持之前状态以及监听F5页面刷新子iframe 而父页面不刷新

    利用layui实现菜单效果时,刷新页面仍回到首页状态,需要 实现iframe子页面刷新父元素不刷新,下面是代码 //刷新时禁用F5的默认事件 $(document).keydown(function ...

  8. centos 下载并安装nodejs

    安装方法1——直接部署 1.首先安装wget ,这个一般都有自带有的,可能已经在系统里安装好了的. yum install -y wget 如果已经安装了可以跳过该步 2.下载nodejs最新的tar ...

  9. 混合云存储组合拳:基于云存储网关与混合云备份的OSS数据备份方案

    前言 阿里云对象存储(OSS)用户众多.很多用户因为业务或者合规性需求,需要对OSS内的数据做备份,无论是线上备份,还是线下备份.用户可以选择使用OSS的开放API,按照业务需求,做数据的备份,也可以 ...

  10. jenkins实现不同角色查看不同视图

    1.安装插件Role-based Authorization Strategy 2.开启插件 系统管理>>>全局安全配置 3.创建角色和用户 4.登陆查看,只能看到travel开头的 ...