トップページに戻る    次の競技プログラミングの問題へ    前の競技プログラミングの問題へ

ABC-070-D Transit Tree Path

■■■問題■■■

N頂点の木が与えられます。

木とはグラフの一種であり、頂点の数をNとすると、辺の数が N-1 本である閉路のない連結グラフです。
i(1 <= i <= N-1) 番目の辺は、頂点aiと頂点biを距離ciで結びます。

また、Q個の質問クエリと整数Kが与えられます。
●j(1 <= j <= Q) 番目の質問クエリでは、頂点xjから頂点Kを経由しつつ、
  頂点yjまで移動する場合の最短経路の距離を求めてください。

■■■入力■■■

N
a1 b1 c1
・
・
・
aN-1 bN-1 cN-1
Q K
x1 y1
・
・
・
xQ yQ

●3 <= N <= 10万
●1 <= ai,bi <= N(1 <= i <= N-1)
●1 <= ci <= 10億(1 <= i <= N-1)
●与えられるグラフは木である
●1 <= Q <= 10万
●1 <= K <= N
●1 <= xj,yj <= N(1 <= j <= Q)
●xj != yj(1 <= j <= Q)
●xj != K,yj != K(1 <= j <= Q)

■■■出力■■■

質問クエリの解答をQ行出力せよ。
j(1 <= j <= Q) 行目には、j番目のクエリの答えを出力せよ。


C#のソース

using System;
using System.Collections.Generic;
using System.Linq;

class Program
{
    static string InputPattern = "Input1";

    static List<string> GetInputList()
    {
        var WillReturn = new List<string>();

        if (InputPattern == "Input1") {
            WillReturn.Add("5");
            WillReturn.Add("1 2 1");
            WillReturn.Add("1 3 1");
            WillReturn.Add("2 4 1");
            WillReturn.Add("3 5 1");
            WillReturn.Add("3 1");
            WillReturn.Add("2 4");
            WillReturn.Add("2 3");
            WillReturn.Add("4 5");
            //3
            //2
            //4
            //与えられた3つの質問クエリに対する最短経路は以下の通りです。
            //●1つ目の質問クエリ: 頂点2 → 頂点1 → 頂点2 → 頂点4 : 距離 1+1+1=3
            //●2つ目の質問クエリ: 頂点2 → 頂点1 → 頂点3 : 距離 1+1=2
            //●3つ目の質問クエリ: 頂点4 → 頂点2 → 頂点1 → 頂点3 → 頂点5 : 距離 1+1+1+1=4
        }
        else if (InputPattern == "Input2") {
            WillReturn.Add("7");
            WillReturn.Add("1 2 1");
            WillReturn.Add("1 3 3");
            WillReturn.Add("1 4 5");
            WillReturn.Add("1 5 7");
            WillReturn.Add("1 6 9");
            WillReturn.Add("1 7 11");
            WillReturn.Add("3 2");
            WillReturn.Add("1 3");
            WillReturn.Add("4 5");
            WillReturn.Add("6 7");
            //5
            //14
            //22
            //質問クエリに対する最短経路は、必ず頂点 K=2 を通過する必要があります。
        }
        else if (InputPattern == "Input3") {
            WillReturn.Add("10");
            WillReturn.Add("1 2 1000000000");
            WillReturn.Add("2 3 1000000000");
            WillReturn.Add("3 4 1000000000");
            WillReturn.Add("4 5 1000000000");
            WillReturn.Add("5 6 1000000000");
            WillReturn.Add("6 7 1000000000");
            WillReturn.Add("7 8 1000000000");
            WillReturn.Add("8 9 1000000000");
            WillReturn.Add("9 10 1000000000");
            WillReturn.Add("1 1");
            WillReturn.Add("9 10");
            //17000000000
        }
        else {
            string wkStr;
            while ((wkStr = Console.ReadLine()) != null) WillReturn.Add(wkStr);
        }
        return WillReturn;
    }

    static int N;

    //辺の情報
    struct EdgeInfoDef
    {
        internal int ToPos;
        internal long Cost;
    }
    //隣接リストで、辺の情報のList[頂点番号]を管理
    static Dictionary<int, List<EdgeInfoDef>> mEdgeInfoDict =
        new Dictionary<int, List<EdgeInfoDef>>();

    static int K;

    //Kからの距離[頂点番号]な配列
    static long[] mKyoriArr;

    static void Main()
    {
        List<string> InputList = GetInputList();

        int[] wkArr = { };
        Action<string> SplitAct = pStr =>
            wkArr = pStr.Split(' ').Select(X => int.Parse(X)).ToArray();

        N = int.Parse(InputList[0]);

        for (int I = 1; I <= N - 1; I++) {
            SplitAct(InputList[I]);
            int A = wkArr[0];
            int B = wkArr[1];
            int C = wkArr[2];

            if (mEdgeInfoDict.ContainsKey(A) == false)
                mEdgeInfoDict.Add(A, new List<EdgeInfoDef>());
            if (mEdgeInfoDict.ContainsKey(B) == false)
                mEdgeInfoDict.Add(B, new List<EdgeInfoDef>());

            mEdgeInfoDict[A].Add(new EdgeInfoDef() { ToPos = B, Cost = C });
            mEdgeInfoDict[B].Add(new EdgeInfoDef() { ToPos = A, Cost = C });
        }

        SplitAct(InputList[N]);
        K = wkArr[1];

        ExecDFS();

        Console.WriteLine("K({0})から各頂点までの距離", K);
        for (int I = 1; I <= N; I++) {
            Console.WriteLine("mKyoriArr[{0}]={1}", I, mKyoriArr[I]);
        }

        //問い合わせの処理
        for (int I = N + 1; I <= InputList.Count - 1; I++) {
            SplitAct(InputList[I]);
            int X = wkArr[0];
            int Y = wkArr[1];

            long Answer = 0;
            Answer += mKyoriArr[X];
            Answer += mKyoriArr[Y];
            Console.WriteLine(Answer);
        }
    }

    struct JyoutaiDef
    {
        internal int CurrPos;
        internal long CostSum;
    }

    //深さ優先探索でKから各頂点までの距離を求める
    //距離[頂点番号]な配列を求める
    static void ExecDFS()
    {
        mKyoriArr = new long[N + 1];

        var Stk = new Stack<JyoutaiDef>();
        JyoutaiDef WillPush;
        WillPush.CurrPos = K;
        WillPush.CostSum = 0;
        mKyoriArr[K] = 0;
        Stk.Push(WillPush);

        var VisitedSet = new HashSet<int>();
        VisitedSet.Add(K);

        while (Stk.Count > 0) {
            JyoutaiDef Popped = Stk.Pop();

            mKyoriArr[Popped.CurrPos] = Popped.CostSum;

            foreach (EdgeInfoDef EachEdgeInfo in mEdgeInfoDict[Popped.CurrPos]) {
                WillPush.CurrPos = EachEdgeInfo.ToPos;
                WillPush.CostSum = Popped.CostSum + EachEdgeInfo.Cost;

                //再訪禁止
                if (VisitedSet.Add(WillPush.CurrPos) == false)
                    continue;

                Stk.Push(WillPush);
            }
        }
    }
}


デバッグ情報付の実行結果

K(1)から各頂点までの距離
mKyoriArr[1]=0
mKyoriArr[2]=1
mKyoriArr[3]=1
mKyoriArr[4]=2
mKyoriArr[5]=2
3
2
4


解説

隣接行列ではなく、隣接リストで辺の情報を管理して、
頂点Kから各頂点までの最短距離を深さ優先探索で求めます。