using System;
using System.Collections.Generic;
using System.Linq;
class Program
{
static string InputPattern = "InputX";
static List<string> GetInputList()
{
var WillReturn = new List<string>();
if (InputPattern == "Input1") {
WillReturn.Add("11 1 4");
WillReturn.Add("1 2");
WillReturn.Add("2 3");
WillReturn.Add("3 5");
WillReturn.Add("3 6");
WillReturn.Add("2 4");
WillReturn.Add("4 7");
WillReturn.Add("1 8");
WillReturn.Add("8 9");
WillReturn.Add("8 10");
WillReturn.Add("9 11");
//1
//1
//2
//1
//3
//3
//2
//2
//3
//3
//4
}
else {
string wkStr;
while ((wkStr = Console.ReadLine()) != null) WillReturn.Add(wkStr);
}
return WillReturn;
}
static int mN;
static int mS;
static int mT;
// 隣接リスト
static Dictionary<int, List<int>> mToNodeListDict = new Dictionary<int, List<int>>();
static void Main()
{
List<string> InputList = GetInputList();
int[] wkArr = { };
Action<string> SplitAct = pStr =>
wkArr = pStr.Split(' ').Select(pX => int.Parse(pX)).ToArray();
SplitAct(InputList[0]);
mN = wkArr[0];
mS = wkArr[1];
mT = wkArr[2];
foreach (string EachStr in InputList.Skip(1)) {
SplitAct(EachStr);
int FromNode = wkArr[0];
int ToNode = wkArr[1];
if (mToNodeListDict.ContainsKey(FromNode) == false) {
mToNodeListDict[FromNode] = new List<int>();
}
if (mToNodeListDict.ContainsKey(ToNode) == false) {
mToNodeListDict[ToNode] = new List<int>();
}
mToNodeListDict[FromNode].Add(ToNode);
mToNodeListDict[ToNode].Add(FromNode);
}
ExecEulerTour(mS, -1, 1);
//foreach (JyoutaiDef_EulerTour EachJyoutai in mEulerTourJyoutaiList) {
// Console.WriteLine("CurrNode={0},Level={1}", EachJyoutai.CurrNode, EachJyoutai.Level);
//}
// 最初に登場するInd[ノード]なDict
var FirstApperIndDict = new Dictionary<int, int>();
// Level[ノード]なDict
var LevelDict = new Dictionary<int, int>();
// スパーステーブルのコンストラクタに使うLevelのList
var LevelList = new List<int>();
for (int I = 0; I <= mEulerTourJyoutaiList.Count - 1; I++) {
int CurrNode = mEulerTourJyoutaiList[I].CurrNode;
if (FirstApperIndDict.ContainsKey(CurrNode) == false) {
FirstApperIndDict[CurrNode] = I;
}
LevelDict[CurrNode] = mEulerTourJyoutaiList[I].Level;
LevelList.Add(mEulerTourJyoutaiList[I].Level);
}
var InsSparseTable = new SparseTable<int>(LevelList, true);
var sb = new System.Text.StringBuilder();
for (int I = 1; I <= mN; I++) {
int Ind1 = FirstApperIndDict[I];
int Ind2 = FirstApperIndDict[mT];
int LeftInd = Math.Min(Ind1, Ind2);
int RightInd = Math.Max(Ind1, Ind2);
int LCALevel = InsSparseTable.Query(LeftInd, RightInd);
int Answer = Math.Abs(LCALevel - LevelDict[I]);
sb.Append(Answer + 1);
sb.AppendLine();
}
Console.Write(sb.ToString());
}
struct JyoutaiDef_EulerTour
{
internal int CurrNode;
internal int Level;
}
// DFSを行い、下記の2通りのタイミングでノードをAddする
// 1 最初の訪問時
// 2 子の部分木を訪問時
static List<JyoutaiDef_EulerTour> mEulerTourJyoutaiList = new List<JyoutaiDef_EulerTour>();
static void ExecEulerTour(int pCurrNode, int pParentNode, int pLevel)
{
JyoutaiDef_EulerTour WillAdd;
WillAdd.CurrNode = pCurrNode;
WillAdd.Level = pLevel;
mEulerTourJyoutaiList.Add(WillAdd);
if (mToNodeListDict.ContainsKey(pCurrNode)) {
foreach (int EachToNode in mToNodeListDict[pCurrNode]) {
if (EachToNode == pParentNode) continue;
ExecEulerTour(EachToNode, pCurrNode, pLevel + 1);
mEulerTourJyoutaiList.Add(WillAdd);
}
}
}
}
#region SparseTable
internal class SparseTable<Type> where Type : IComparable<Type>
{
// 最小値か最大値[2の指数][開始Ind]なArr
private Type[][] mMinOrMaxJagArr;
// Log2の値[2べき値] なDict
private Dictionary<long, long> mLogDict = new Dictionary<long, long>();
// 対応する2べき値[クエリの区間長]
private long[] mBeki2Arr;
// 最小値を取得するか?
private bool mIsMin = false;
// コンストラクタ1(配列が引数)
internal SparseTable(Type[] pInitArr, bool pIsMin)
{
mIsMin = pIsMin;
long Sisuu = 0;
long Beki2 = 1;
while (true) {
mLogDict[Beki2] = Sisuu;
Beki2 *= 2;
if (Beki2 > pInitArr.Length) {
break;
}
Sisuu++;
}
mMinOrMaxJagArr = new Type[Sisuu + 1][];
foreach (var EachPair in mLogDict) {
long CurrBeki2 = EachPair.Key;
long CurrSisuu = EachPair.Value;
long CurrUB = pInitArr.GetUpperBound(0) - CurrBeki2 + 1;
mMinOrMaxJagArr[CurrSisuu] = new Type[CurrUB + 1];
}
// 長さ1の分を初期化
for (long I = 0; I <= pInitArr.GetUpperBound(0); I++) {
mMinOrMaxJagArr[0][I] = pInitArr[I];
}
// コンストラクタのヘルパメソッド
ConstructorHelper();
// 対応する2べき値[クエリの区間長]を設定
SetBeki2Arr(pInitArr.Length);
}
// コンストラクタ2(Listが引数)
internal SparseTable(List<Type> pInitList, bool pIsMin)
{
mIsMin = pIsMin;
long Sisuu = 0;
long Beki2 = 1;
while (true) {
mLogDict[Beki2] = Sisuu;
Beki2 *= 2;
if (Beki2 > pInitList.Count) {
break;
}
Sisuu++;
}
mMinOrMaxJagArr = new Type[Sisuu + 1][];
foreach (var EachPair in mLogDict) {
long CurrBeki2 = EachPair.Key;
long CurrSisuu = EachPair.Value;
long CurrUB = pInitList.Count - 1 - CurrBeki2 + 1;
mMinOrMaxJagArr[CurrSisuu] = new Type[CurrUB + 1];
}
// 長さ1の分を初期化
for (int I = 0; I <= pInitList.Count - 1; I++) {
mMinOrMaxJagArr[0][I] = pInitList[I];
}
// コンストラクタのヘルパメソッド
ConstructorHelper();
// 対応する2べき値[クエリの区間長]を設定
SetBeki2Arr(pInitList.Count);
}
// コンストラクタ3(列挙が引数)
internal SparseTable(IEnumerable<Type> pInitEnum, bool pIsMin)
{
mIsMin = pIsMin;
Type[] InitArr = pInitEnum.ToArray();
long Sisuu = 0;
long Beki2 = 1;
while (true) {
mLogDict[Beki2] = Sisuu;
Beki2 *= 2;
if (Beki2 > InitArr.Length) {
break;
}
Sisuu++;
}
mMinOrMaxJagArr = new Type[Sisuu + 1][];
foreach (var EachPair in mLogDict) {
long CurrBeki2 = EachPair.Key;
long CurrSisuu = EachPair.Value;
long CurrUB = InitArr.GetUpperBound(0) - CurrBeki2 + 1;
mMinOrMaxJagArr[CurrSisuu] = new Type[CurrUB + 1];
}
// 長さ1の分を初期化
for (int I = 0; I <= InitArr.GetUpperBound(0); I++) {
mMinOrMaxJagArr[0][I] = InitArr[I];
}
// コンストラクタのヘルパメソッド
ConstructorHelper();
// 対応する2べき値[クエリの区間長]を設定
SetBeki2Arr(InitArr.Length);
}
// 対応する2べき値[クエリの区間長]を設定
private void SetBeki2Arr(long pDataLen)
{
long[] Beki2Arr = mLogDict.Keys.ToArray();
mBeki2Arr = new long[pDataLen + 1];
long CurrBeki2 = 1;
for (long I = 0; I <= pDataLen; I++) {
mBeki2Arr[I] = CurrBeki2;
if (mLogDict.ContainsKey(I)) {
CurrBeki2 = I;
}
}
}
// コンストラクタのヘルパメソッド
private void ConstructorHelper()
{
for (long LoopLength = 2; LoopLength < long.MaxValue; LoopLength *= 2) {
if (mLogDict.ContainsKey(LoopLength) == false) break;
long HalfRange = LoopLength / 2;
Type[] CurrArr = mMinOrMaxJagArr[mLogDict[HalfRange]];
Type[] SendArr = mMinOrMaxJagArr[mLogDict[LoopLength]];
for (long I = 0; I <= SendArr.GetUpperBound(0); I++) {
long StaInd1 = I;
long EndInd1 = I + HalfRange - 1;
long StaInd2 = EndInd1 + 1;
long EndInd2 = StaInd2 + HalfRange - 1;
Type Val1 = CurrArr[I];
Type Val2 = CurrArr[StaInd2];
int CompResult = Val1.CompareTo(Val2);
if (mIsMin) {
SendArr[I] = (CompResult >= 1 ? Val2 : Val1);
}
else {
SendArr[I] = (CompResult >= 1 ? Val1 : Val2);
}
}
}
}
// 閉区間 [L,R]の最小値または最大値を求める
internal Type Query(long pL, long pR)
{
// 区間を内包する2べき値を使用
long RangeLen = pR - pL + 1;
long Beki2 = mBeki2Arr[RangeLen];
long Sisuu = mLogDict[Beki2];
Type Val1 = mMinOrMaxJagArr[Sisuu][pL];
Type Val2 = mMinOrMaxJagArr[Sisuu][pR - Beki2 + 1];
int CompResult = Val1.CompareTo(Val2);
if (mIsMin) {
return (CompResult >= 1 ? Val2 : Val1);
}
return (CompResult >= 1 ? Val1 : Val2);
}
}
#endregion