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. Spring-Security (学习记录二)--修改为自己的登录页面

    目录 1.修改spring-security.xml配置文件 2.增加login.jsp页面 3.重启项目即可看到效果 1.修改spring-security.xml配置文件 <!-- auto ...

  2. elasticsearch的基本用法(转载)

    本文出自:http://blog.csdn.net/feelig/article/details/8499614 最大的特点:  1. 数据库的 database, 就是  index  2. 数据库 ...

  3. 函数的属性和方法, apply和call的区别及bind的使用

    ==>我的新博客中 http://www.suanliutudousi.com/2017/08/27/%E5%87%BD%E6%95%B0%E7%9A%84%E5%B1%9E%E6%80%A7% ...

  4. vue中excal表格的导入和导出

    注意:vue中要实现表格的导入与导出,首先要install两个依赖, npm install -S file-saver xlsx  和  npm install -D script-loader.其 ...

  5. 【POJ】1860 Currency Exchange

    真是气skr人..没把d[]换成double...de了一上午的bug// 记得用G++提交啊 题目链接:http://poj.org/problem?id=1860 题意:告诉你n个点,m条路.起始 ...

  6. Linux queue.h之TAILQ队列分析

    转自 这两天想看看memcached的实现,所以先学习了libevent,使用起来还是比较简单的,其实是对select/poll/kqueue等的封装,学习libevent过程中又遇到了linux下队 ...

  7. [JZOJ1901] 【2010集训队出题】光棱坦克

    题目 题目大意 给你个平面上的一堆点,问序列\({p_i}\)的个数. 满足\(y_{p_{i-1}}>y_{p_i}\)并且\(x_{p_i}\)在\(x_{p_i-1}\)和\(x_{p_i ...

  8. thinkphp 运算符

    我们可以对模板输出使用运算符,包括对“+”“ –” “*” “/”和“%”的支持. 大理石平台厂家 例如: 运算符 使用示例 + {$a+$b} - {$a-$b} * {$a*$b} / {$a/$ ...

  9. thinkphp 自动加载

    在3.2中,基本上无需手动加载类库文件,你可以很方便的完成自动加载. 命名空间自动加载 系统可以通过类的命名空间自动定位到类库文件,例如: 我们定义了一个类 Org\Util\Auth 类: name ...

  10. 巧用CSS3的calc()宽度计算做响应模式布局

    今天浏览这个http://www.sitepoint.com站时,因为好奇看了下人家写的代码,结果发现了这行代码, 于是就研究了一下,calc()从字面我们可以把他理解为一个函数function.其实 ...