zoukankan      html  css  js  c++  java
  • 【Splay】【启发式合并】hdu6133 Army Formations

    题意:给你一颗树,每个结点的儿子数不超过2。每个结点有一个权值,一个结点的代价被定义为将其子树中所有结点的权值放入数组排序后,每个权值乘以其下标的和。让你计算所有结点的代价。

    二叉树的条件没有用到。

    每个结点开一个Splay,从叶子往上启发式合并上去,可以先bfs一遍确定合并顺序。每一次将Splay大小较小的结点的权值全提取出来塞到较大的里面。

    由于权值可能重复出现,所以每个结点记个cnt。

    答案统计的时候,就将那个刚塞进去的旋到根,然后答案加上左子树的权值和,再加上(右子树的权值的个数+该结点的cnt)*该结点的权值。

    然后将较大儿子的Splay的根丢给父亲。

    不必要进行内存回收,通过计算,所开的空间只要达到nlogn即可,实际上100w足够了。

    #include<cstdio>
    #include<set>
    #include<queue>
    #include<cstring>
    #include<algorithm>
    using namespace std;
    typedef long long ll;
    const int maxn=1000000;
    ll sum[maxn];
    int fa[maxn],val[maxn],c[maxn][2],root,tot,siz[maxn],cnt[maxn];
    void Maintain(int x)
    {
        siz[x]=siz[c[x][0]]+siz[c[x][1]]+cnt[x];
        sum[x]=sum[c[x][0]]+sum[c[x][1]]+(ll)val[x]*(ll)cnt[x];
    }
    void NewNode(int &x,int Fa,int key)
    {
        x=++tot;
        fa[x]=Fa;
        c[x][0]=c[x][1]=0;
        val[x]=key;
        siz[x]=cnt[x]=1;
    }
    void Rotate(int x,bool flag)
    {
        int y=fa[x];
        c[y][!flag]=c[x][flag];
        fa[c[x][flag]]=y;
        if(fa[y]){
            c[fa[y]][c[fa[y]][1]==y]=x;
        }
        fa[x]=fa[y];
        c[x][flag]=y;
        fa[y]=x;
        Maintain(y);
        Maintain(x);
    }
    void Splay(int &root,int x,int goal)
    {
        if(!x){
            return;
        }
        int y;
        while((y=fa[x])!=goal){
            if(fa[y]==goal){
                Rotate(x,c[y][0]==x);
            }
            else{
                if((c[y][0]==x)==(c[fa[y]][0]==y)){
                    Rotate(y,c[fa[y]][0]==y);
                }
                else{
                    Rotate(x,c[y][0]==x);
                    y=fa[x];
                }
                Rotate(x,c[y][0]==x);
            }
        }
        Maintain(x);
        if(!goal){
            root=x;
        }
    }
    int Find(int key,int x)
    {
        while(c[x][val[x]<key]){
            if(val[x]==key){
                return x;
            }
            x=c[x][val[x]<key];
        }
        return x;
    }
    void Insert(int &root,int key)
    {
        int x=Find(key,root);
        if(val[x]==key){
            ++cnt[x];
            Splay(root,x,0);
            return;
        }
        NewNode(c[x][val[x]<key],x,key);
        Splay(root,c[x][val[x]<key],0);
    }
     
     
     
     
    int roots[100005];
    ll anss[100005];
    bool vis[100005];
    int e,first[100005],nex[200005],v[200005];
    int dep[100005];
    void AddEdge(int U,int V){
        v[++e]=V;
        nex[e]=first[U];
        first[U]=e;
    }
    int T,n,a[100005],b[100005];
    bool cmp(const int &a,const int &b){
        return siz[roots[a]]>siz[roots[b]];
    }
    int sons[100005],BI;
    void dfs(int U){
        for(int i=1;i<=cnt[U];++i){
            Insert(roots[sons[1]],val[U]);
            int X=Find(val[U],roots[sons[1]]);
            Splay(roots[sons[1]],X,0);
            if(c[X][0]){
                anss[BI]+=sum[c[X][0]];
            }
            anss[BI]+=(ll)(cnt[X]+(c[X][1] ? siz[c[X][1]] : 0))*(ll)val[U];
        }
        if(c[U][0]){
            dfs(c[U][0]);
        }
        if(c[U][1]){
            dfs(c[U][1]);
        }
    }
    int main(){
    //  freopen("a.in","r",stdin);
    //  freopen("a.out","w",stdout);
        scanf("%d",&T);
        int x,y;
        for(;T;--T){
            memset(first,0,sizeof(first));
            e=0;
            scanf("%d",&n);
            for(int i=1;i<=n;++i){
                scanf("%d",&a[i]);
            }
            for(int i=1;i<n;++i){
                scanf("%d%d",&x,&y);
                AddEdge(x,y);
                AddEdge(y,x);
            }
            memset(vis,0,sizeof(vis));
            memset(dep,0,sizeof(dep));
            memset(anss,0,sizeof(anss));
            queue<int>q;
            q.push(1);
            int t=0;
            while(!q.empty()){
                int U=q.front(); q.pop(); vis[U]=1;
                b[++t]=U;
                for(int i=first[U];i;i=nex[i]){
                    if(!vis[v[i]]){
                        dep[v[i]]=dep[U]+1;
                        q.push(v[i]);
                    }
                }
            }
            for(int i=t;i>=1;--i){
                BI=b[i];
                int dir_son=0;
                for(int j=first[b[i]];j;j=nex[j]){
                    if(dep[v[j]]>dep[b[i]]){
                        sons[++dir_son]=v[j];
                    }
                }
                if(!dir_son){
                    NewNode(roots[b[i]],0,a[b[i]]);
                    anss[b[i]]=a[b[i]];
                    continue;
                }
                sort(sons+1,sons+dir_son+1,cmp);
                anss[b[i]]=anss[sons[1]];
                for(int j=2;j<=dir_son;++j){
                    dfs(roots[sons[j]]);
                }
                Insert(roots[sons[1]],a[b[i]]);
                int X=Find(a[b[i]],roots[sons[1]]);
                Splay(roots[sons[1]],X,0);
                if(c[X][0]){
                    anss[BI]+=sum[c[X][0]];
                }
                anss[BI]+=(ll)(cnt[X]+(c[X][1] ? siz[c[X][1]] : 0))*(ll)a[b[i]];
                roots[b[i]]=roots[sons[1]];
            }
            for(int i=1;i<=n;++i){
                printf("%lld ",anss[i]);
            }
            puts("");
            memset(sum,0,sizeof(ll)*(tot+1));
            memset(fa,0,sizeof(int)*(tot+1));
            memset(val,0,sizeof(int)*(tot+1));
            for(int i=0;i<=tot;++i){
                c[i][0]=c[i][1]=0;
            }
            memset(siz,0,sizeof(int)*(tot+1));
            memset(cnt,0,sizeof(int)*(tot+1));
            memset(roots,0,sizeof(int)*(tot+1));
    //      printf("%d
    ",tot);
            tot=0;
        }
        return 0;
    }
  • 相关阅读:
    sqlite3 多线程和锁 ,优化插入速度及性能优化
    Android Touch事件传递机制通俗讲解
    26. Remove Duplicates from Sorted Array----Array----Easy----20160927
    88. Merge Sorted Array----Array----Easy----20160926
    @121. Best Time to Buy and Sell Stock----Array----Easy---20160926
    88. Merge Sorted Array----Array----Easy----20160925
    189. Rotate Array----Array----Easy----20160925
    169. Majority Element----Array----Easy----20160925
    219. Contains Duplicate II 超时
    c++Map容器 详解
  • 原文地址:https://www.cnblogs.com/autsky-jadek/p/7384891.html
Copyright © 2011-2022 走看看