天天看點

hdu 3974 Assign the task

題目連結如下所示:

http://acm.hdu.edu.cn/showproblem.php?pid=3974

這個線段樹特别的地方在于用一個dfs建立了區間到樹節點的映射,于是每個節點代表區間,一切操作轉化為線段樹的操作。

代碼如下所示:

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<ctime>
#include<vector>
#include<algorithm>
#include<cstring>
#include<set>
#include<map>
#include<queue>
#include<stack>
#include<string>
#include<cmath>
#include<climits>

using namespace std;
#define lson l,mid,rt<<1
#define rson mid+1,r,rt<<1|1
typedef long long ll;
const int MAXN=5e4+5;
int l_t[MAXN],r_t[MAXN];
int degree[MAXN];
ll tree[MAXN<<3];
vector<int> edge[MAXN];
int N,tot;

void dfs(int rt){
    l_t[rt]=++tot;
    for (int i = 0; i < edge[rt].size(); ++i) {
        dfs(edge[rt][i]);
    }
    r_t[rt]=++tot;
}

void build(int l,int r,int rt){
    tree[rt]=-1;
    if (l==r) return;
    int mid=(l+r)>>1;
    build(lson);
    build(rson);
}

void push_down(int rt){
    if (tree[rt]!=-1){
        tree[rt<<1]=tree[rt];
        tree[rt<<1|1]=tree[rt];
        tree[rt]=-1;
    }
}

void update(int L,int R,int val,int l,int r,int rt){
    if (L<=l && r<=R){
        tree[rt]=val;
        return;
    }
    push_down(rt);
    int mid=(l+r)>>1;
    if (L<=mid){
        update(L,R,val,lson);
    }
    if (R>mid){
        update(L,R,val,rson);
    }
}

int query(int L,int l,int r,int rt){
    if (l==r && L==l) return tree[rt];
    push_down(rt);
    int mid=(l+r)>>1;
    if (L<=mid){
        return query(L,lson);
    } else {
        return query(L,rson);
    }
}

int main(){
    scanf("%d", &N);
    int n,u,v,m,x,y;
    int kase=0;
    while (N--){
        tot = 0;
        memset(degree, 0, sizeof(degree));
        for (int i = 0; i < MAXN; ++i) {
            edge[i].clear();
        }
        scanf("%d", &n);
        for (int i = 1; i < n; ++i) {
            scanf("%d %d",&u,&v);
            edge[v].push_back(u);
            degree[u]++;
        }
        int root=-1;
        for (int i = 1; i <= n; ++i) {
            if (!degree[i]){
                root=i;
                break;
            }
        }
        dfs(root);
        build(1,2*n,1);
        printf("Case #%d:\n",++kase);
        scanf("%d",&m);
        while (m--){
            char op[2];
            scanf("%s",op);
            if (op[0]=='C'){
                scanf("%d",&x);
                printf("%d\n", query(l_t[x],1,2*n,1));
            }else{
                scanf("%d %d",&x,&y);
                update(l_t[x],r_t[x],y,1,2*n,1);
            }
        }
    }
    return 0;
}
           

繼續閱讀