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. iOS13适配/黑暗模式的适配/KVC访问私有属性/模态弹窗ViewController 默认样式改变 /LaunchImage即将废弃/蓝牙的权限申请/推送Device Token适配/UIKit 控件变化/StatusBar新增样式

    目录 1. KVC访问私有属性 2. 模态弹窗ViewController 默认样式改变 3. 黑暗模式的适配 4. LaunchImage即将废弃 5. 新增一直使用蓝牙的权限申请 6. Sign ...

  2. CPUID 指令的使用

    使用 CPUID 指令可以从 processor 厂商里获得关于 processor 的详细信息,CPUID 指令是从 Intel 486 处理器以后开始加入支持. 1. 检测处理器是否支持 cpui ...

  3. Invalid bound statement (not found): com.my.demo.mapper.UserMapper.getAll

    网上的方法全试了,配置全对,很奇怪. 最后一查,竟然忘记把mapper保存为xml格式. 记录一下.

  4. c++ 11新特性学习1

    static_assert 静态断言,特点是编译期的断言检查 assert 运行时期的断言检查 二者参数用法相同

  5. AtCoder ABC 130F Minimum Bounding Box

    题目链接:https://atcoder.jp/contests/abc130/tasks/abc130_f 题目大意 给定地图上 N 个点的坐标和移动方向,它们会以每秒 1 个单位的速度移动,设 A ...

  6. kubeadm 安装k8s

    环境要求: 机器名 ip地址 cpu和内存要求 kubernetes-master 10.0.0.11 2c2g(关闭swap) kubernetes-node1 10.0.0.12 2c2g(关闭s ...

  7. 利用IK分词器,自定义分词规则

    IK分词源码下载地址:https://code.google.com/p/ik-analyzer/downloads/list lucene源码下载地址:http://www.eu.apache.or ...

  8. API文档管理工具

    系统庞大之后,前后端分离开发,前端调用后端提供的接口,请求协议一般是 HTTP,数据格式一般是 JSON.后台只负责数据的提供和计算,而完全不处理展现逻辑和样式:前端则负责拿到数据,组织数据并展现的工 ...

  9. duboo注解使用详解

    一.背景 随着互联网的发展,网站应用的规模不断扩大,常规的垂直应用架构已无法应对,分布式服务架构以及流动计算架构势在必行. 当越来越的的接口与实现类的增加后,duboo的xml配置会越来越多,为了防止 ...

  10. 关于js私钥加密公钥解密的问题

    博客荒废很久了,最近遇到一个问题,看网上的说明比较少,所以写下来给大家一个参考 一般来说rsa算法都是使用公钥加密,私钥解密,或者私钥签名,公钥验签.但总有特别的时候会想要用私钥加密,公钥解密,但是j ...