zoukankan      html  css  js  c++  java
  • SPOJ375Query on a tree I(树剖+线段树)(询问边)

    ιYou are given a tree (an acyclic undirected connected graph) with N nodes, and edges numbered 1, 2, 3...N-1.

    We will ask you to perfrom some instructions of the following form:

    • CHANGE i ti : change the cost of the i-th edge to ti
      or
    • QUERY a b : ask for the maximum edge cost on the path from node a to node b

    Input

    The first line of input contains an integer t, the number of test cases (t <= 20). t test cases follow.

    For each test case:

    • In the first line there is an integer N (N <= 10000),
    • In the next N-1 lines, the i-th line describes the i-th edge: a line with three integers a b c denotes an edge between ab of cost c (c <= 1000000),
    • The next lines contain instructions "CHANGE i ti" or "QUERY a b",
    • The end of each test case is signified by the string "DONE".

    There is one blank line between successive tests.

    Output

    For each "QUERY" operation, write one integer representing its result.

    Example

    Input:
    1
    
    3
    1 2 1
    2 3 2
    QUERY 1 2
    CHANGE 1 3
    QUERY 1 2
    DONE
    
    Output:
    1
    3

    题意:

    对于一棵树,有如下两种操作;

    • 询问u,v两点间的最短路上权值最大的边。
    • 更改某条边的权值。

    思路:

    树剖基础题,大家都会,而我刚刚学,所以就不啰嗦了。

    如果是更新点,查询点,可能好写点。而这里是对边进行处理,写半天把自己搅混了。。。

    代码里是这样处理的:

    • 对于每条边的两个顶点u,v,距离root远的是u,不然交换即可。假设这条边是i,用tid[u]表示这条边在线段树中对应的位置。即tid x表示x和父亲这条边在线段树中的位置。用时间戳tim++保证了root一下的x才有对应的tid;
    #include<cstdio>
    #include<cstdlib>
    #include<cstring>
    #include<iostream>
    #include<algorithm>
    #include<cstring>
    const int maxn=20010;
    using namespace std;
    int e[maxn][3];
    int dpt[maxn],son[maxn],fa[maxn];
    int Laxt[maxn],Next[maxn],To[maxn],cnt;
    int sz[maxn],tid[maxn],top[maxn],tim;//树剖 
    int Max[maxn],Rank[maxn],n;//线段树 
    //tid[v]表示v与其父亲节点的连边在线段树中的位置 即time'id
    struct TC
    {
        int lowbit(int x) { return x&(-x);}
        void init()
        {
            cnt=1; tim=0;
            memset(Laxt,0,sizeof(Laxt));
            memset(Max,0,sizeof(Max));
            memset(son,0,sizeof(son));
        }
        void add_edge(int u,int v)
        {
            Next[++cnt]=Laxt[u]; 
            Laxt[u]=cnt; To[cnt]=v;
        }
        int dfs1(int u,int pre)//得到dpt,fa,son,sz 
        {
            fa[u]=pre;dpt[u]=dpt[pre]+1;sz[u]=1;
            for(int i=Laxt[u];i;i=Next[i]){
                int v=To[i];
                if(v!=pre){
                    dfs1(v,u); sz[u]+=sz[v];
                    if(!son[u]||sz[v]>sz[son[u]]) son[u]=v;
                }
            }
        }
        void dfs2(int u,int Top)//得到边在线段树中的位置。 
        {
            top[u]=Top; tid[u]=tim++; Rank[tid[u]]=u;//是tim++,而不是++tim,从而把root排除,弱保存点是后者 
            if(!son[u]) return ;
            dfs2(son[u],Top); 
            for(int i=Laxt[u];i;i=Next[i]){
                int v=To[i];
                if(v!=fa[u]&&v!=son[u]) {
                     dfs2(v,v);//这里不再是Top了,换链了。 
                }
            }
        }
        int update(int Now,int L,int R,int pos,int num)
        {
            if(L==pos&&R==pos) return Max[Now]=num; 
            int Mid=(L+R)>>1;//下面注意不要丢掉return,WA了两发。不然多写个push_up也可以。 
            if(pos<=Mid) return Max[Now]=max(update(Now<<1,L,Mid,pos,num),Max[Now<<1|1]);
            else return Max[Now]=max(update(Now<<1|1,Mid+1,R,pos,num),Max[Now<<1]);
        }
        void Make_tree()
        {
            int u,v;
            for(int i=1;i<n;i++){
                scanf("%d%d%d",&e[i][0],&e[i][1],&e[i][2]);
                add_edge(e[i][0],e[i][1]); 
                add_edge(e[i][1],e[i][0]);
            }
            dfs1(1,0);  
            dfs2(1,1);
            for(int i=1;i<n; i++){//单点更新建树 
                if(dpt[e[i][0]]>dpt[e[i][1]]) swap(e[i][0],e[i][1]);
                update(1,1,n-1,tid[e[i][1]],e[i][2]);
            }
        } 
        int get_max(int Now,int L,int R,int l,int r)
        {
            if(L>=l&&R<=r) return Max[Now];
            int Mid=(L+R)>>1;
            if(Mid>=r) return get_max(Now<<1,L,Mid,l,r);
            else if(Mid+1<=l) return get_max(Now<<1|1,Mid+1,R,l,r);
            else return max(get_max(Now<<1,L,Mid,l,r),get_max(Now<<1|1,Mid+1,R,l,r));
        }
        void Query()
        {
            int u,v; scanf("%d%d",&u,&v);
            int f1=top[u],f2=top[v];
            int ans=0;
            while(f1 != f2){
               if(dpt[f1] < dpt[f2]){
                  swap(f1,f2); swap(u,v);
               }
               ans=max(ans,get_max(1,1,n-1,tid[f1],tid[u]));
               u=fa[f1]; f1=top[u];
            }
            if(u!=v){
                if(dpt[u]>dpt[v]) swap(u,v);
                ans=max(ans,get_max(1,1,n-1,tid[son[u]],tid[v])); 
            }
            printf("%d
    ",ans);
        }
        void Change()
        {
            int u,w; scanf("%d%d",&u,&w);
            update(1,1,n-1,tid[e[u][1]],w);
        }
    }Tc;
    int main()
    {
        int T; char c[10];
        scanf("%d",&T);
        while(T--){
            Tc.init();
            scanf("%d",&n);
            Tc.Make_tree();
            while(~scanf("%s",&c)){
                if(c[0]=='D') break;
                else if(c[0]=='Q') Tc.Query();
                else Tc.Change();
            }
        } return 0;
    }
  • 相关阅读:
    未能加载包“Microsoft SQL Server Data Tools”
    SharePoint 客户端对象模型共用ClientContext的坑
    安装VisualStudio 2015 x64 中文企业版失败
    Could not load file or assembly 'Microsoft.SqlServer.Management.Sdk.Sfc, Version=11.0.0.0 系统找不到指定的文件。
    为Sharepoint 2010 批量创建SharePoint测试用户
    User Profile Service Application 配置同步连接时,报 MOSS MA not found
    SharePoint 2010 系统账户没完全控制权限了
    百度编辑器 UEditor 报错汇总
    更改SharePoint 2007/2010/2013 Web 应用程序端口号
    SharePoint 2013 报:网站在改进过程中处于只读状态,对此给您带来的不便,我们深表歉意
  • 原文地址:https://www.cnblogs.com/hua-dong/p/8059856.html
Copyright © 2011-2022 走看看