题目

源地址:

http://codeforces.com/problemset/problem/20/C

理解

实际上题目不难,但是我们都捣鼓了很久。原因是我们根本就没有掌握这种算法,导致连一个输出路径都搞得这么蛋疼。使用邻接表来存储每一个节点,每一个节点都自带一个指针指向下一个节点(可以自己使用数组模拟),最后的结果倒过来输出即可。

代码


#define MAXN 100000*4+10

const ll maxd = 1E13;
int v[MAXN],w[MAXN],next[MAXN],pre[MAXN],res[MAXN];
int first[MAXN],inq[MAXN],e;
ll d[MAXN];
int m,n;
int x,y,z;


void addeage(int x,int y,int z)
{
    v[e]=y;
    w[e]=z;
    next[e]=first[x];
    first[x]=e;
    e++;
}

void spfa(int s)
{
    queue<int> q;
    for(int i =0; i<MAXN; i++)
        d[i]=maxd;
    d[s]=0;
    inq[s]=1;
    q.push(s);
    while(q.empty()==false)
    {
        int u = q.front();
        q.pop();
        inq[u]=0;
        for(int i =first[u]; i!=-1; i=next[i])
        {
            if(d[v[i]]>d[u]+w[i])
            {
                d[v[i]]=(d[u]+w[i]);
                pre[v[i]]=u;
                if(inq[v[i]]==0)
                {
                    q.push(v[i]);
                    inq[v[i]]=1;
                }
            }
        }
    }
}

void init()
{
    for(int i=0; i<MAXN; i++)
    {
        first[i]=-1;
    }
    e=0;
    scanf("%d%d",&m,&n);
    while(n--)
    {
        scanf("%d%d%d",&x,&y,&z);
        addeage(x,y,z);
        addeage(y,x,z);
    }
}


int main(int argc, char const *argv[])
{
    init();
    spfa(1);
    if(d[m]==maxd)
        printf("-1");
    else
    {
        int now=m;
        int cnt=0;
        while(now!=1)
        {
            res[cnt++] = now;
            now = pre[now];
        }
        res[cnt++] = 1;
        for(int i = cnt-1; i >= 0; i--)
            printf("%d ",res[i]);
    }
    return 0;
}

更新日志

  • 2014年11月26日 已AC。