absi2011's Blog & Daily Life.

全新的开始       我要省选翻盘       I wanna AK in 高考\化学       自此,生无可恋
CF 292E 解题报告
100286 B 解题报告 & 碎碎念

321 C 解题报告

absi2011 posted @ Feb 14, 2016 04:44:46 PM in 刷题记录 with tags CF 树链剖分 , 474 阅读

感谢似水流年的翻译,题目链接:http://codeforces.com/contest/321/problem/C

题意:

给你一颗树

要你构造一个方案,给每个点标上一个字母

使得任意两个B之间都至少通过一个A,任意两个C之间都通过至少一个B或A...任意两个Z之前都通过至少一个A~Y

感谢Pascal Endlife教我树的重心求法

做法:每次找树的重心,然后把它设置为当前权值最大的点

之后把整个树剖开来,这个点拿走,变成一颗颗子树,再分别这么做

这样做下去肯定有解

树的重心求法:同树链剖分的dfs1,最后再求一下它上面的大小(总大小-子树大小)即可

代码:

#include<set>
#include<map>
#include<list>
#include<queue>
#include<stack>
#include<string>
#include<time.h>
#include<math.h>
#include<memory>
#include<vector>
#include<bitset>
#include<fstream>
#include<stdio.h>
#include<utility>
#include<sstream>
#include<string.h>
#include<iostream>
#include<stdlib.h>
#include<algorithm>
using namespace std;
char ans[100005];
int size[100005];
int visit[100005];
int best_ans;
int best_val;
struct edge
{
    int y;
    edge * next;
};
edge * li[100005];
edge * new_edge()
{
    static edge a[200005];
    static int top=0;
    return &a[top++];
}
void inserts(int x,int y)
{
    edge * t=new_edge();
    t->y=y;
    t->next=li[x];
    li[x]=t;
}
void insert_edge(int x,int y)
{
    inserts(x,y);
    inserts(y,x);
}
void dfs1(int x)
{
    size[x]=1;
    visit[x]=true;
    edge * t;
    for (t=li[x];t!=0;t=t->next)
    {
        if (ans[t->y]!='\0') continue;
        if (visit[t->y]) continue;
        dfs1(t->y);
        size[x]+=size[t->y];
    }
}
void dfs2(int x,int sumsize)
{
    edge * t;
    visit[x]=true;
    int max_ans=sumsize-size[x];
    for (t=li[x];t!=0;t=t->next)
    {
        if (ans[t->y]!='\0') continue;
        if (visit[t->y]) continue;
        dfs2(t->y,sumsize);
        max_ans=max(max_ans,size[t->y]);
    }
    if (max_ans<best_val)
    {
        best_val=max_ans;
        best_ans=x;
    }
}
int main()
{
    #ifdef absi2011
    freopen("input.txt","r",stdin);
    freopen("output.txt","w",stdout);
    #endif
    int n;
    scanf("%d",&n);
    int i;
    for (i=0;i<n;i++)
    {
        int x,y;
        scanf("%d%d",&x,&y);
        x--;
        y--;
        insert_edge(x,y);
    }
    memset(ans,'\0',sizeof(ans));
    for (i=0;i<26;i++)
    {
        memset(size,0,sizeof(size));
        memset(visit,0,sizeof(visit));
        int j;
        for (j=0;j<n;j++)
        {
            if (ans[j]!='\0') continue;
            if (visit[j]) continue;
            dfs1(j);
        }
        memset(visit,0,sizeof(visit));
        for (j=0;j<n;j++)
        {
            if (ans[j]!='\0') continue;
            if (visit[j]) continue;
            best_ans=-1;
            best_val=10000000;
            dfs2(j,size[j]);
            ans[best_ans]=i+'A';
        }
    }
    for (i=0;i<n;i++)
    {
        printf("%c ",ans[i]);
    }
    printf("\n");
    return 0;
}

登录 *


loading captcha image...
(输入验证码)
or Ctrl+Enter