CodeForces - 1209F Koala and Notebook(拆边+BFS)

题意:给定一个n个点m条边的无向图,边权分别为1-m,从起点1出发,每经过一条边就把边权以字符串的形式加入末尾,求到达其他每个点的最小字符串(长度不同的短的更小,否则字典序小的更小)。

思路很巧妙,将每个边按照边权的位数拆成若干条虚边+若干个虚点,然后以1为起点进行BFS,边权相同的放在一起处理,这样就能保证每个点第一次到达的路径是字符串最小的路径了。

边权排序可以用基排,因为只能取0-9,常数很小。

注意各种细节。

#include<bits/stdc++.h>
using namespace std;
typedef long long ll;
const int N=1e6+10,inf=0x3f3f3f3f,mod=1e9+7;
int hd[N],ne,n,m,dp[N],tot,buf[N],C[10];
struct E {int u,v,c,nxt;} e[N<<1];
void addedge(int u,int v,int c) {e[ne]= {u,v,c,hd[u]},hd[u]=ne++;}
queue<int> q;
vector<int> vec,vec2;
int upd(int u,int x) {
    if(!~dp[u]) {dp[u]=x,q.push(u); return 1;}
    return 0;
}
void bfs() {
    memset(dp,-1,sizeof dp);
    while(q.size())q.pop();
    upd(1,0),q.push(-1);
    while(q.size()) {
        vec.clear();
        for(; q.front()!=-1; q.pop())vec.push_back(q.front());
        q.pop();
        vec2.clear();
        for(int u:vec)for(int i=hd[u]; ~i; i=e[i].nxt)vec2.push_back(i);
        for(int i=0; i<=9; ++i)C[i]=0;
        for(int i=0; i<vec2.size(); ++i)++C[e[vec2[i]].c];
        for(int i=1; i<=9; ++i)C[i]+=C[i-1];
        for(int i=0; i<vec2.size(); ++i)buf[--C[e[vec2[i]].c]]=vec2[i];
        for(int i=0,j,k; i<vec2.size(); i=j) {
            for(j=i; j<vec2.size()&&e[buf[j]].c==e[buf[i]].c; ++j);
            int f=0;
            for(k=i; k<j; ++k)if(upd(e[buf[k]].v,((ll)dp[e[buf[k]].u]*10+e[buf[k]].c)%mod))f=1;
            if(f)q.push(-1);
        }
    }
}
int main() {
    memset(hd,-1,sizeof hd),ne=0;
    scanf("%d%d",&n,&m);
    tot=n;
    for(int i=1; i<=m; ++i) {
        int u,v;
        scanf("%d%d",&u,&v);
        int c=i;
        if(c/10) {
            addedge(++tot,u,c%10),addedge(++tot,v,c%10),c/=10;
            for(; c/10; c/=10)++tot,addedge(tot,tot-2,c%10),++tot,addedge(tot,tot-2,c%10);
            addedge(v,tot-1,c),addedge(u,tot,c);
        } else addedge(u,v,c),addedge(v,u,c);
    }
    bfs();
    for(int i=2; i<=n; ++i)printf("%d\n",dp[i]);
    return 0;
}

 

posted @ 2019-10-04 19:51  jrltx  阅读(169)  评论(0编辑  收藏  举报