SPOJ10707 COT2-Count on a tree II

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;
} 

转载于:https://www.cnblogs.com/huibixiaoxing/p/11254289.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值