<?php 
 
 
 
    SUBWAY =>  'едешь на метро' , 
    FOOT =>  'идешь пешком' , 
    BUS =>  'едешь на автобусе' 
) ; 
 
$startPoint  =  'pet' ;  // Петроградская 
$endPoint  =  'nov' ;  // Новая Голландия 
 
    'pet'  =>  'ст. м. Петроградская' , 
    'chk'  =>  'ст. м. Чкаловская' , 
    'gor'  =>  'ст. м. Горьковская' , 
    'spo'  =>  'ст. м. Спортивная' , 
    'vas'  =>  'ст. м. Василеостровская' , 
    'kre'  =>  'Петропавловская крепость' , 
    'let'  =>  'Летний сад' , 
    'dvo'  =>  'Дворцовая площадь' , 
    'isa'  =>  'Исакиевский собор' , 
    'nov'  =>  'Новая Голландия' , 
    'ras'  =>  'Дом Раскольникова' , 
    'gos'  =>  'Гостиный Двор' , 
    'sen'  =>  'Сенная Площадь' , 
    'vla'  =>  'ст. м. Владимирская' , 
    'vit'  =>  'Витебский вокзал' , 
    'teh'  =>  'Технолоческий Институт' 
) ; 
 
        'chk'  =>  canGet( 10 ,  BUS) , 
        'gor'  =>  canGet( 3 ,  SUBWAY) 
    ) , 
 
        'pet'  =>  canGet( 10 ,  BUS) , 
        'spo'  =>  canGet( 3 ,  SUBWAY) 
    ) , 
 
        'pet'  =>  canGet( 3 ,  BUS) , 
        'kre'  =>  canGet( 5 ,  FOOT) , 
        'gos'  =>  canGet( 6 ,  SUBWAY) 
    ) , 
 
        'chk'  =>  canGet( 3 ,  SUBWAY) , 
        'vas'  =>  canGet( 10 ,  BUS) , 
        'sen'  =>  canGet( 7 ,  SUBWAY) 
    ) , 
 
        'spo'  =>  canGet( 10 ,  BUS) , 
        'gos'  =>  canGet( 7 ,  SUBWAY) , 
        'nov'  =>  canGet( 11 ,  FOOT) 
    ) , 
 
        'gor'  =>  canGet( 5 ,  FOOT) 
    ) , 
 
        'dvo'  =>  canGet( 6 ,  FOOT) , 
        'gos'  =>  canGet( 7 ,  FOOT) 
    ) , 
 
        'isa'  =>  canGet( 6 ,  FOOT) , 
        'gos'  =>  canGet( 6 ,  FOOT) , 
        'let'  =>  canGet( 6 ,  FOOT) 
    ) , 
 
        'dvo'  =>  canGet( 6 ,  FOOT) , 
        'nov'  =>  canGet( 5 ,  FOOT) 
    ) , 
 
        'vas'  =>  canGet( 11 ,  FOOT) , 
        'isa'  =>  canGet( 5 ,  FOOT) , 
        'ras'  =>  canGet( 7 ,  BUS) 
    ) , 
 
        'nov'  =>  canGet( 7 ,  BUS) , 
        'sen'  =>  canGet( 3 ,  FOOT) 
    ) , 
 
        'vas'  =>  canGet( 7 ,  SUBWAY) , 
        'sen'  =>  canGet( 3 ,  SUBWAY) , 
        'dvo'  =>  canGet( 6 ,  FOOT) , 
        'gor'  =>  canGet( 6 ,  SUBWAY) , 
        'let'  =>  canGet( 7 ,  FOOT) , 
        'vla'  =>  canGet( 7 ,  FOOT) 
    ) , 
 
        'ras'  =>  canGet( 3 ,  FOOT) , 
        'spo'  =>  canGet( 7 ,  SUBWAY) , 
        'gos'  =>  canGet( 3 ,  SUBWAY) , 
        'vla'  =>  canGet( 4 ,  SUBWAY) , 
        'vit'  =>  canGet( 2 ,  SUBWAY) , 
        'teh'  =>  canGet( 3 ,  SUBWAY) 
    ) , 
 
        'sen'  =>  canGet( 4 ,  SUBWAY) , 
        'gos'  =>  canGet( 7 ,  FOOT) , 
        'vit'  =>  canGet( 3 ,  SUBWAY) 
    ) , 
 
        'sen'  =>  canGet( 2 ,  SUBWAY) , 
        'teh'  =>  canGet( 2 ,  SUBWAY) , 
        'vla'  =>  canGet( 3 ,  SUBWAY) 
    ) , 
 
        'sen'  =>  canGet( 3 ,  SUBWAY) , 
        'vit'  =>  canGet( 2 ,  SUBWAY) 
    ) 
) ; 
 
/* Чтобы не писать много раз array('time' => ..., 'by' => ...), используем функцию. 
    «canGet» переводится как «можно попасть» */ 
function  canGet( $time ,  $byWhat ) 
{ 
    return  array ( 'time'  =>  $time ,  'by'  =>  $byWhat ) ;  } 
 
# Возвращает ближайший узел 
function  lowestWeightNode( $weightOfActiveNodes ) 
{ 
    $lowestWeight  =  INF; 
    $lowestNode  =  null ; 
    foreach  ( $weightOfActiveNodes  as  $node  =>  $weightNode )  { 
        if  ( $lowestWeight  >  $weightNode )  { 
            $lowestWeight  =  $weightNode ; 
            $lowestNode  =  $node ; 
        } 
    } 
    return  $lowestNode ; 
} 
 
function  searchForAWay( $startPoint ,  $endPoint ,  $paths ,  $transportName ,  $pointNames ) 
{ 
    $weightOfActiveNodes  =  array_fill_keys ( $point ,  INF
) ;  // Массив с весами узлов      $parentsNode  =  array_fill_keys ( $point ,  '' ) ;;  // Массив с родительскикими узлами      $weightOfProcessedKnots  =  [ ] ;  // Массив для обработаннхы узлов 
 
    $weightOfActiveNodes [ $startPoint ]  =  0 ;  // Стартовой точке присваивается 0, для того, алгоритм поиска пути начался с неё 
    $node  =  lowestWeightNode( $weightOfActiveNodes ) ;  // Находит узел с наименьшим весом 
 
    while  ( $node  !=  $endPoint )  { 
        foreach  ( $paths [ $node ]  as  $neighbor  =>  $timeAndBy )  { 
                $weightNeighbor  =  $weightOfActiveNodes [ $node ]  +  $timeAndBy [ 'time' ] ;  // весСоседа = рассматриваемый узел + путь до соседа 
                if  ( $weightOfActiveNodes [ $neighbor ]  >  $weightNeighbor )  {  // Если текущий вес узла (рассматриваемого соседа) больше чем , новый найденный 
                    $parentsNode [ $neighbor ]  =  $node ; 
                    $weightOfActiveNodes [ $neighbor ]  =  $weightNeighbor ; 
                } 
            } 
        } 
        $weightOfProcessedKnots [ $node ]  =  $weightOfActiveNodes [ $node ] ; 
        unset ( $weightOfActiveNodes [ $node ] ) ;          $node  =  lowestWeightNode( $weightOfActiveNodes ) ; 
    } 
 
    $commonPath  =  [ ] ;  // Кратчайший, требуемый путь 
    $finishPoint  =  $endPoint ; 
    while  ( $finishPoint  !=  $startPoint )  { 
        $commonPath [ $finishPoint ]  =  $parentsNode [ $finishPoint ] ; 
        $finishPoint  =  $parentsNode [ $finishPoint ] ; 
    } 
 
    #Вывод пути 
    $totalTime  =  0 ; 
    foreach  ( $commonPath  as  $startPoint  =>  $FinishPoint )  { 
        $wayOfMovement  =  $transportName [ $paths [ $startPoint ] [ $FinishPoint ] [ 'by' ] ] ; 
        echo  "Из неё {$wayOfMovement}  до точки {$pointNames [$startPoint ]} {$paths [$startPoint ][$FinishPoint ]['time']} мин."  .  PHP_EOL; 
        $totalTime  +=  $paths [ $startPoint ] [ $FinishPoint ] [ 'time' ] ; 
    } 
    echo  "В итоге ты попадешь в точку {$pointNames [$endPoint ]} за {$totalTime}  минут" ; 
} 
 
searchForAWay( $startPoint ,  $endPoint ,  $paths ,  $transportName ,  $pointNames ) ; 
 
				<?php

error_reporting(-1);

define('SUBWAY', 'sub');
define('FOOT', 'foot');
define('BUS', 'bus');

$transportName = array(
    SUBWAY => 'едешь на метро',
    FOOT => 'идешь пешком',
    BUS => 'едешь на автобусе'
);

$startPoint = 'pet'; // Петроградская
$endPoint = 'nov'; // Новая Голландия

$pointNames = array(
    'pet' => 'ст. м. Петроградская',
    'chk' => 'ст. м. Чкаловская',
    'gor' => 'ст. м. Горьковская',
    'spo' => 'ст. м. Спортивная',
    'vas' => 'ст. м. Василеостровская',
    'kre' => 'Петропавловская крепость',
    'let' => 'Летний сад',
    'dvo' => 'Дворцовая площадь',
    'isa' => 'Исакиевский собор',
    'nov' => 'Новая Голландия',
    'ras' => 'Дом Раскольникова',
    'gos' => 'Гостиный Двор',
    'sen' => 'Сенная Площадь',
    'vla' => 'ст. м. Владимирская',
    'vit' => 'Витебский вокзал',
    'teh' => 'Технолоческий Институт'
);

$paths = array(
    'pet' => array(
        'chk' => canGet(10, BUS),
        'gor' => canGet(3, SUBWAY)
    ),

    'chk' => array(
        'pet' => canGet(10, BUS),
        'spo' => canGet(3, SUBWAY)
    ),

    'gor' => array(
        'pet' => canGet(3, BUS),
        'kre' => canGet(5, FOOT),
        'gos' => canGet(6, SUBWAY)
    ),

    'spo' => array(
        'chk' => canGet(3, SUBWAY),
        'vas' => canGet(10, BUS),
        'sen' => canGet(7, SUBWAY)
    ),

    'vas' => array(
        'spo' => canGet(10, BUS),
        'gos' => canGet(7, SUBWAY),
        'nov' => canGet(11, FOOT)
    ),

    'kre' => array(
        'gor' => canGet(5, FOOT)
    ),

    'let' => array(
        'dvo' => canGet(6, FOOT),
        'gos' => canGet(7, FOOT)
    ),

    'dvo' => array(
        'isa' => canGet(6, FOOT),
        'gos' => canGet(6, FOOT),
        'let' => canGet(6, FOOT)
    ),

    'isa' => array(
        'dvo' => canGet(6, FOOT),
        'nov' => canGet(5, FOOT)
    ),

    'nov' => array(
        'vas' => canGet(11, FOOT),
        'isa' => canGet(5, FOOT),
        'ras' => canGet(7, BUS)
    ),

    'ras' => array(
        'nov' => canGet(7, BUS),
        'sen' => canGet(3, FOOT)
    ),

    'gos' => array(
        'vas' => canGet(7, SUBWAY),
        'sen' => canGet(3, SUBWAY),
        'dvo' => canGet(6, FOOT),
        'gor' => canGet(6, SUBWAY),
        'let' => canGet(7, FOOT),
        'vla' => canGet(7, FOOT)
    ),

    'sen' => array(
        'ras' => canGet(3, FOOT),
        'spo' => canGet(7, SUBWAY),
        'gos' => canGet(3, SUBWAY),
        'vla' => canGet(4, SUBWAY),
        'vit' => canGet(2, SUBWAY),
        'teh' => canGet(3, SUBWAY)
    ),

    'vla' => array(
        'sen' => canGet(4, SUBWAY),
        'gos' => canGet(7, FOOT),
        'vit' => canGet(3, SUBWAY)
    ),

    'vit' => array(
        'sen' => canGet(2, SUBWAY),
        'teh' => canGet(2, SUBWAY),
        'vla' => canGet(3, SUBWAY)
    ),

    'teh' => array(
        'sen' => canGet(3, SUBWAY),
        'vit' => canGet(2, SUBWAY)
    )
);

/* Чтобы не писать много раз array('time' => ..., 'by' => ...), используем функцию.
    «canGet» переводится как «можно попасть» */
function canGet($time, $byWhat)
{
    return array('time' => $time, 'by' => $byWhat);
}

# Возвращает ближайший узел
function lowestWeightNode($weightOfActiveNodes)
{
    $lowestWeight = INF;
    $lowestNode = null;
    foreach ($weightOfActiveNodes as $node => $weightNode) {
        if ($lowestWeight > $weightNode) {
            $lowestWeight = $weightNode;
            $lowestNode = $node;
        }
    }
    return $lowestNode;
}

function searchForAWay($startPoint, $endPoint, $paths, $transportName, $pointNames)
{
    $point = array_keys($paths);
    $weightOfActiveNodes = array_fill_keys($point, INF); // Массив с весами узлов
    $parentsNode = array_fill_keys($point, '');; // Массив с родительскикими узлами
    $weightOfProcessedKnots = []; // Массив для обработаннхы узлов

    $weightOfActiveNodes[$startPoint] = 0; // Стартовой точке присваивается 0, для того, алгоритм поиска пути начался с неё
    $node = lowestWeightNode($weightOfActiveNodes); // Находит узел с наименьшим весом

    while ($node != $endPoint) {
        foreach ($paths[$node] as $neighbor => $timeAndBy) {
            if (array_key_exists($neighbor, $weightOfActiveNodes)) { // 
                $weightNeighbor = $weightOfActiveNodes[$node] + $timeAndBy['time']; // весСоседа = рассматриваемый узел + путь до соседа
                if ($weightOfActiveNodes[$neighbor] > $weightNeighbor) { // Если текущий вес узла (рассматриваемого соседа) больше чем , новый найденный
                    $parentsNode[$neighbor] = $node;
                    $weightOfActiveNodes[$neighbor] = $weightNeighbor;
                }
            }
        }
        $weightOfProcessedKnots[$node] = $weightOfActiveNodes[$node];
        unset($weightOfActiveNodes[$node]);
        $node = lowestWeightNode($weightOfActiveNodes);
    }

    $commonPath = []; // Кратчайший, требуемый путь
    $finishPoint = $endPoint;
    while ($finishPoint != $startPoint) {
        $commonPath[$finishPoint] = $parentsNode[$finishPoint];
        $finishPoint = $parentsNode[$finishPoint];
    }
    $commonPath = array_reverse($commonPath);

    #Вывод пути
    $totalTime = 0;
    foreach ($commonPath as $startPoint => $FinishPoint) {
        $wayOfMovement = $transportName[$paths[$startPoint][$FinishPoint]['by']];
        echo "Из неё {$wayOfMovement} до точки {$pointNames[$startPoint]} {$paths[$startPoint][$FinishPoint]['time']} мин." . PHP_EOL;
        $totalTime += $paths[$startPoint][$FinishPoint]['time'];
    }
    echo "В итоге ты попадешь в точку {$pointNames[$endPoint]} за {$totalTime} минут";
}

searchForAWay($startPoint, $endPoint, $paths, $transportName, $pointNames);