Задача из Моего Собеседования в Amazon - Поиск в Ширину

แชร์
ฝัง
  • เผยแพร่เมื่อ 25 พ.ย. 2024

ความคิดเห็น • 254

  • @sashalukin
    @sashalukin  7 หลายเดือนก่อน

    Создал Telegram канал, в котором рассказываю о жизни в Лондоне, работе в Google и подготовке к собеседованиям. Подписывайтесь: t.me/saschalukin

  • @denys_barkhatov
    @denys_barkhatov 8 หลายเดือนก่อน +91

    С анимацией отлично получилось.
    Очень интересно смотреть твои разборы задач + тембр и подача приятные.
    Если позволяет время - пиши побольше видео.

  • @misterbob9827
    @misterbob9827 8 หลายเดือนก่อน +16

    Никогда даже не хотел понимать задачи с шахматной доской , после просмотра понял алгоритм 💪🏾

  • @neo3248
    @neo3248 8 หลายเดือนก่อน +168

    Увидимся через полтора года!

    • @rockkley9159
      @rockkley9159 8 หลายเดือนก่อน +3

      Почему полтора года?

    • @Mr.Bellamy
      @Mr.Bellamy 8 หลายเดือนก่อน

      @@rockkley9159 потому что видео на этом канале выходят по особому алгоритму

    • @krestolit6999
      @krestolit6999 8 หลายเดือนก่อน +3

      Ну как успехи?

    • @opalev
      @opalev 8 หลายเดือนก่อน +9

      полтора года прошли уже, чо как?

    • @dedpudge1374
      @dedpudge1374 8 หลายเดือนก่อน

      ​@@opalevгде они прошли

  • @amirilifee
    @amirilifee 8 หลายเดือนก่อน +14

    Саша анимация вообще супер!!! Видео было настолько интересным что ни секунды не промотал. Спасибо за такое понятное и четкое объяснение

  • @khetagabramov5124
    @khetagabramov5124 8 หลายเดือนก่อน +25

    Для всех, кто пишет что в алгоритме ошибка, так как мы смотрим только на предыдущий слой и теряем информацию о предыдущих слоях:
    На самом деле ошибки нет, вероятно комментаторы путают с алгоритмом DFS или вариантом BFS на очереди (где нужно запоминать все посещенные вершины).
    Почему этот алгоритм корректен? Если предположить что на текущем шаге мы попытаемся пойти из ячейки X в ячейку Y, в которой уже были, но которая лежит не в предыдущем слое, то значит она лежит в предпредыдущем или более ранних слоях. Но тогда мы бы перешли по этому ребру Y - X в тот момент, когда рассматривали эту клетку Y и в этом случае X лежало бы не в текущем слое, а в следующем за слоем, в котором лежит Y. Но в предположении X лежит именно в текущем, а Y лежит не в предыдущем. Значит предположение неверно, а значит достаточно рассматривать для клеток, которые мы хотим посетить, лежат ли они в предыдущем слое. И этого будет достаточно.

    • @Abingusus
      @Abingusus 8 หลายเดือนก่อน +2

      Более простое объяснение: предыдущий слой это как бы затычка для пути назад. А поскольку это поиск в ширину, то мы эти затычки ставим для всех возможных путей назад, то есть как не старайся не получится "обойти" предыдущий слой со стороны.
      Аналогия такая: два длинных коридора с дверьми, которые иногда пересекаются. Ты и твой друг идете одновременно от начала корридоров, и закрываете за собой каждую дверь. Так вот ты не сможешь в месте пересечения коридора пойти назад, т.к. твой друг уже закрыл за собой предыдущуб дверь. При этом не важно сколько до этого дверей было закрыто, главное что он последнюю закрыл в том месте, где вы встретились, поэтому ты не сможешь через его коридор пойти назад, и ты будешь вынужден пойти дальше вперед

    • @krolikrodjer8879
      @krolikrodjer8879 8 หลายเดือนก่อน +2

      @@Abingususчет объяснение намного запутаннее, чем первый коммент)

    • @МехДи-с2т
      @МехДи-с2т 8 หลายเดือนก่อน

      Спасибо за объяснение. А для тех кто считает, что лучше один раз увидеть. Посмотрите на доску на которой Саша показал уже 3-й слой (4:08). Где-нибудь можно попасть с клетки 3-го слоя на клетку слоя 0 или 1.

  • @olegorlov3329
    @olegorlov3329 8 หลายเดือนก่อน +17

    Если брать очень большое поле, то для общего случая нахождения кратчайшего пути стоит добавит разбивку на сектора.
    В случае с конем можно сделать еще проще, первые ходы должны делаться с учетом кратчайшей дистанции, и только начиная с дистанции в 4 клетки выполнять обозначенный автором кульбит

    • @Abingusus
      @Abingusus 8 หลายเดือนก่อน +2

      Вот я тоже думаю, обход поля n*n это O(n^2), но кажется можно за линейную сложность. Типа мы можем просчитать окрестность исходной точки, потом из каждой точки окрестности сделать простой алгоритм, который примерно
      1. Находит два необходимых вида шагов O(1)
      2. Считает необходимое колво шагов O(1) k
      3. Находит точное количества шагов для шага типа1 и типа2 (линейный поиск O(n), т.к суммарно шагов тип1+тип2 = k точно меньше 2n), и если ответ нашелся, значит это будет тип1+тип2+шаг окрестности. Если не нашелся, значит забиваем на эту точку окрестности
      4. Повторяем для каждой точки окрестности и ищем минимальное среди ответов O(c), с количество точек в окрестности
      Итого у на фиксированная окрестность а количество шагов растет линейно, значит сложность лин поиска тоже растет линейно.
      Также заметим, что порядок шагов тип1 и тип2 не важен, так что тут все ок.

    • @gleb2971
      @gleb2971 8 หลายเดือนก่อน

      Да, с динамическим программированием получится эффективнее.

  • @GSergeiA
    @GSergeiA 8 หลายเดือนก่อน +17

    Мощно ты запарился за анимации!) Но получилось явно очень наглядно и понятно!

  • @_Smai1e_
    @_Smai1e_ 8 หลายเดือนก่อน +16

    Здесь можно (даже бы я сказал нужно) использовать алгоритм А*, а за эвристическию функцию можно взять манхеттенское расстояние от коня до конечной клетки

    • @PimiTree
      @PimiTree 8 หลายเดือนก่อน +1

      Не знал что у єтого есть название, но решал бы тоже "через растояние" и ходы, которые его сокращают

    • @ЕгорПичугин-у5м
      @ЕгорПичугин-у5м 6 หลายเดือนก่อน +1

      Кстати, не уверен, что А* сработает в данном случае, так как алгоритм выберет самую ближайшую клетку к конечной клетке, но так как конь ходит буквой Г он же может крутиться несколько раз вокруг конечной клетки что бы в неё попасть. А соседняя клетка (которая будет иметь большее манхэтоновское расстояние) попадёт в конечную клетку раньше, хоть эвристика в ней изначально была больше. Я не ради прикопаться, мне тоже сразу в голову пришёл этот алгоритм, просто интересно так это или нет.

    • @smokyOoOwizard
      @smokyOoOwizard 5 หลายเดือนก่อน

      @@ЕгорПичугин-у5мконь бы не крутился вокруг цели. Он бы просто игнорировал уже пройденные ( найденые ходы ). Если с первой попытки он был не нашел путь, то начал бы обходить ближайшие. Но даже так это было бы куда эффективнее по памяти, чем делать поиск в ширину, так как мы не хранили бы явно плохие ходы

  • @jpxfrd787
    @jpxfrd787 8 หลายเดือนก่อน +19

    Долго тебя не было, рад новому видео!

  • @laremin_developer
    @laremin_developer 8 หลายเดือนก่อน +6

    Выпускай чаще пожалуйста, ты лучший мой мотиватор )))

  • @antonkuzin8544
    @antonkuzin8544 8 หลายเดือนก่อน +42

    Привет! Заметил пару неточностей.
    1. Ты говоришь, что не посещаешь уже посещенные клетки. Но это утверждение верно только для предыдущего слоя - в коде ты никак не помечаешь те клетки, что были посещены ещё раньше.
    2. Формально это можно назвать обходом графа в ширину, но мне кажется, это решение методом динамического программирования. А если попытаться оптимизироавть решение и не проверять заведомо более длинные пути (а ограничиться только квадратом (0, 0) (n, n)), то это уже точно не будет обходом графа в ширину(так как мы не посещаем все вершины).
    3. Решение данной конкретной задачи обходом графа в ширину довольно неэффективно - сложность получается ~(2n)^2*Log(n), тогда как методом динамического программирования мы можем обойтись n^2, перебрав каждую вершину урезанного квадрата лишь единожды. n это большая по модулю координата целевой точки.
    Upd: попытавшись набросать решение методом динамического программирования, понял, что у меня не получается) Конь ходит слишком сложно, чтобы за один проход можно было заполнить массив, не возвращаясь к уже посещённым клеткам. А вот в обходе графа ограничить поле всё равно было бы нелишним.

    • @xdxd3631
      @xdxd3631 8 หลายเดือนก่อน

      так может обозначишь их?

    • @MrFunStuf88
      @MrFunStuf88 8 หลายเดือนก่อน

      Если входной массив можно редактировать, то одной из оптимизаций будет простая "покраска" значения

    • @Arkham_nine
      @Arkham_nine 8 หลายเดือนก่อน +1

      > если попытаться оптимизироавть решение и не проверять заведомо более длинные пути (а ограничиться только квадратом (0, 0) (n, n)), то это...
      ....не будет работать. Например конь (0,0) выход (1,1)
      А вообще, мне кажется тут есть математическое решение.

    • @khetagabramov5124
      @khetagabramov5124 8 หลายเดือนก่อน +7

      1) В этом и суть решения - можно проверять только предыдущий слой, нельзя попытаться пойти из слоя №3 НЕ в предыдущий слой №2 или следующий слой №4 (иначе бы слои формировались по-другому), поэтому не имеет смысла проверять все посещенные вершины, а только посещенные на последнем слое. Как раз таки поэтому это алгоритм обхода в глубину на двух списках (более классическое решение - одна очередь).
      2) Метод BFS (Breadth-first search) как ни странно использует идею динамического программирования. Одно использует, а не заменяет другое. Вы ведь не скажете "Ну я бы не стал доезжать на работу на Мерседесе, я бы предпочел просто доехать на машине." Поиск в ширину реализуется на динамическом программировании. Посещать не все вершины - не значит не обходить граф, но оптимизации действительно здесь приемлемы (можете еще почитать про итеративный поиск в глубину iterative deepening depth-first search).
      3) Тут используем хеш таблицу, а не дерево поиска, поэтому время работы алгоритма O(n^2). Но вы все еще правы, что BFS реализуется через идею динамического программирования, просто обход ячеек при рекурентном пересчете соответствует обходу графа.

    • @antonkuzin8544
      @antonkuzin8544 8 หลายเดือนก่อน +2

      @@Arkham_nine хорошо, считаем до [n + 2, n + 2] :)

  • @user-lk8n0fgjk
    @user-lk8n0fgjk 8 หลายเดือนก่อน +3

    Спасибо! Отличное видео, интересная задача! Ждем новых роликов.

  • @EvilYarik
    @EvilYarik 6 หลายเดือนก่อน +1

    Тут экспоненциальный рост клеток с каждым слоем и симметрия 4-го порядка, стоила взять координаты цели по модулю и начать с двух ходов 1, 2 и 2, 1 на первом слое. Так-же можно игнорировать все клетки с отрицательными координатами, это сэкономит память.

  • @alexandre2024ds
    @alexandre2024ds 7 หลายเดือนก่อน

    Спасибо за задачу! Очень подробно объяснил)

  • @danieldeveloper2614
    @danieldeveloper2614 8 หลายเดือนก่อน +4

    Congrats Alex! Good explanation.
    Here is a point you can deal better with:
    You were java dev, and you use java terminology in your verbal explanations, but code is written in python. So little fellas can easily miss those word about hashed collection.
    Hope you'll continue doing videos like this)

    • @widny31
      @widny31 8 หลายเดือนก่อน +7

      онглечанен, ставит скобку после текста 😂

  • @itananas
    @itananas 6 หลายเดือนก่อน

    Вау, очень крутое и понятное объяснение. Анимация - огонь

  • @ДаниилМонахов-р8ч
    @ДаниилМонахов-р8ч 8 หลายเดือนก่อน +19

    Вообще в классическом БФС используется очередь (Queue), а не некие уровни.
    1. Запихиваешь в очередь первую клетку.
    2. Извлекаешь.
    3. Находишь всех соседей, в нашем случае восемь возможных ходов коня.
    4. Фильтруешь уже посещённые.
    5. Запихиваешь в очередь.
    6. Гото 2 до победы.
    Если смотреть только на previous_layer, то потеряется информация о посещённых клетках на предыдущих ходах.

    • @rasimbot
      @rasimbot 8 หลายเดือนก่อน +1

      Уровень в данном случае это синоним глубины. Какие к этому могут быть претензии?

    • @КамильБадрутдинов-р8щ
      @КамильБадрутдинов-р8щ 8 หลายเดือนก่อน +3

      @@rasimbot ну bfs как будто именно для поиска путей используется, а не расстояний, хотя его и можно упростить до поиска минимального расстояния. + как и сказано в видосе, bfs запускается на графе, хоть шахматная доска и представима в виде графа, сам граф в этом решении не строится и вряд ли можно говорить о bfs, хотя и оч похоже. Например, если эту задачку чуть изменить и просить искать не кратчайшее расстояние, а кратчайший путь (именно вывести все посещенные клетки по порядку), то решение из видоса как будто не позволит этого сделать, с bfs получилось бы (ну мне так кажется). Я не с целью надушнить, если что, если это решение без построение графа можно под вышеописанную задачу изменить, буду рад услышать)

    • @vanel9933
      @vanel9933 8 หลายเดือนก่อน

      @@КамильБадрутдинов-р8щ учи теорию графов лучше)

    • @КамильБадрутдинов-р8щ
      @КамильБадрутдинов-р8щ 8 หลายเดือนก่อน

      @@vanel9933 так а что учить то подскажи)

  • @semenkovalev4988
    @semenkovalev4988 6 หลายเดือนก่อน

    Сначала конь на клетке 1:1 (зачем-то), потом уже на логичной 0:0. Ну и можно оптимизировать конкретный кейс с учетом осей симметрии доски и обработать только 1/8 всех клеток.

  • @МихаилЧ-д2х
    @МихаилЧ-д2х 7 หลายเดือนก่อน

    Обычная задача на волновой алгоритм. По предоставленному решению. Нет контроля выхода за границу доски. Также проверяем попадание в клетку на предыдущем уровне, но не на уровнях ранее, для данного контроля можно использовать обычный двумерный массив описывающий доску и хранящий уровни посещения в каждой клетке

  • @flashbackenigma517
    @flashbackenigma517 8 หลายเดือนก่อน +4

    Блин, смотрю каждое твое видео. Спасибо за контент. Тоже мечтаю попасть в google

    • @Chel1k7
      @Chel1k7 8 หลายเดือนก่อน

      мечтать мало, нужно работать над собой, чтоб туда попасть

    • @n1ret
      @n1ret 6 หลายเดือนก่อน

      эта задача очень лёгкая

  • @madplayer5
    @madplayer5 8 หลายเดือนก่อน +4

    Анимация наглядная, круто!

  • @yohohowowowo9471
    @yohohowowowo9471 6 หลายเดือนก่อน

    Можно ещё параллельно вычислять дорогу от коня к точке и от точки к коню ровно до тех пор покуда не найдётся первое пересечение. Действий суммарно будет больше, но зато массив будет меньше...хотел сказать я, но потом обнаружил одно геометрическое свойство: сумма длин окружностей, пересекающихся в одной точке равна длине окружности с радиусом, равным расстоянию между центрами этих окружностей)))

  • @НиколайВоронин-о7ъ
    @НиколайВоронин-о7ъ 6 หลายเดือนก่อน

    Спасибо за bfs!
    Но в аналитическом решении сложность О(1).
    N = (lnl + lml)/3 ± 4. Начальные координаты коня (0,0), конечные (n, m). Если рассмотреть как ходить в конце (варианты), то формула будет без погрешностей.

  • @МаксимМаликов-г4ъ
    @МаксимМаликов-г4ъ 7 หลายเดือนก่อน

    Решение автора работает похоже на работу ИИ, он также связывает всё послойно, и между всеми элементами одного слоя и другого присутствует связь, таким образом происходит имитация синапсов, что помогает ему осмыслить любые возможные варианты.

  • @ventice11o
    @ventice11o 8 หลายเดือนก่อน +2

    функция moves более надежно но абсолютно нечитабельно записывается в данном виде:
    from itertools import permutations, product
    return [tuple(em * es for em, es in zip(m,s)) for m, s in product(permutations([1, 2]), product(*permutations([1, -1])))]

    • @widny31
      @widny31 8 หลายเดือนก่อน

      по-моему, вот ЭТО более нечитабельно. Функция moves выглядит куда проще и нагляднее

    • @ventice11o
      @ventice11o 8 หลายเดือนก่อน

      @@widny31 ну да, я и написал, что хардкод в данном случае читабельнее. просто показал, как в Питоне выражается "по одной оси на два, по другой -- на один во всех возможный направлениях" посредством функций из itertools

  • @kirill.leontovich
    @kirill.leontovich 8 หลายเดือนก่อน +2

    Во-первых нормализовать данные. Стартовая клетка пусть будет (0,0), а координаты цели во-первых возьмем по модулю, что не изменит количество шагов, а перенесет цель в верхний правый квадрант заеркально осей, и затем если Х > Y поменяем их местами, чтобы поместить цель выше диагонали (0,0) - (N,N). Для квадрата 5х5 проще всего создать хэш-таблицу, выйдет всего 14 элементов. Если цель дальше 5 клеток то "прыгаем" к ней двигая доску, или вычитая (2,1) из координат цели, добавляя к счетчику 1. Если на очередной итерации цель ушла левее оси Y (отрицательный X), либо "упала" под диагональ (Y < X), то вычитаем не (2,1), а либо (2, -1), либо (1, 2) соответственно. Физический смысл легко увидеть если нарисовать каждую из ситуаций. Как только на очередной итерации мы попали в квадрат 5х5 к цели, снова нормализуем координаты и берем значение из таблицы, добавляем к счетчику итераций и получаем итоговое.
    Некрасивая таблица из 14 значений с лихвой компенсируется сложностью О(n) и нулевым выделением памяти на каждой итерации. Описывать сложнее, чем написать код.

    • @maxshibanov818
      @maxshibanov818 5 หลายเดือนก่อน

      Я тоже так решил. Я бы взял разность между координатами по модулю и вычитал (2, 1), (1, 2), пока это возможно, а затем конь либо достигнет цели, либо доберется до любой клетки рядом за 2 хода.

  • @AlexanderSavchenko91
    @AlexanderSavchenko91 6 หลายเดือนก่อน

    Блин! точно! это же графы! спасибо большое!

  • @mats.7684
    @mats.7684 8 หลายเดือนก่อน

    у меня в школе была задача посчитать за какое минимальное число шагов конь пройдет по каждой клетке поля по 1 разу. И всё это на паскале с отрисовкой доски и прочего)

  • @Poli.Pavlovich
    @Poli.Pavlovich หลายเดือนก่อน

    Анимация супер! С помощью чего она сделана? Объяснение тоже класс, после него стала меньше бояться задач на шахматной доске. Немного сложно было воспринимать объяснение кода под музыку. Отдельное спасибо за питон!

  • @rusfungame
    @rusfungame 6 หลายเดือนก่อน

    У меня 4 года опыта в программировании, 3 года коммерции, чувствую себя таким джуном) Старший разработчик - джун.

  • @ambassadorcherta7615
    @ambassadorcherta7615 8 หลายเดือนก่อน

    Сразу в голову пришла идея через обход в ширину как только сказал про хеш-таблицу ( чтобы в ней хранить позиции, в которых мы были ). На код варсе это 3-4 тир задач.

  • @ventice11o
    @ventice11o 8 หลายเดือนก่อน +2

    в случае коня есть много места для эвристической оптимизации BFS, чтобы не идти в заведомо ложном направлении

  • @RusLSystems
    @RusLSystems 8 หลายเดือนก่อน

    Вычисляем первый круг, а дальше копипастим весь массив с наложением в нужную сторону.
    Если требуется заполнение всей доски или идет непопадание, возможен банальный брутфорс,
    либо от последней позиции, либо вообще всех возможных комбинаций массива.

  • @AnthonyMarchenko
    @AnthonyMarchenko 17 วันที่ผ่านมา

    Спасибо, очень крутая анимация, такую в Keynote просто сделаешь, так?

  • @МихаилГагин-л5с
    @МихаилГагин-л5с 8 หลายเดือนก่อน +1

    Если наше поле заранее известного размера, то надо использовать обычный массив, а не хеш таблицы. И второе, поиск в ширину выглядит красиво, когда наша цель находиться в нескольких шагах, а когда до неё шагов 30, то мы умрём прежде чем поиск в ширину найдёт путь.

    • @MMI-memyselfi
      @MMI-memyselfi 5 หลายเดือนก่อน

      задача - проверить понимание поиска в ширину.. дальше на интервью можно было бы спросить "а давай оптимизируем?"

  • @BigRock379
    @BigRock379 7 หลายเดือนก่อน

    Делай, пожалуйста, больше видео по алгоритмам

  • @КириллТонковид
    @КириллТонковид 6 หลายเดือนก่อน

    я бы решал через алгебру, но нужно обработать крайние случаи и написать костыли. А так если люди не против и нужно что-то сделать за короткое время то и так можно.

  • @pretcb
    @pretcb 8 หลายเดือนก่อน

    Спасибо за видео. На курсах по изучению Пайтон решал эту же задачу:)) интересно посмотреть твое решение. Лайк

  • @finch694
    @finch694 5 หลายเดือนก่อน

    я бы решал это волновым алгоритмом (алгоритм Ли) без той части где путь восстанавливается. моё решение может быть чуть хуже по памяти, потому что нужно было бы создать двумерный массив с запасом по три клетки от каждой координаты старта и финиша. за то весьма наглядно бы получилось)) номер фронта волны который запишется в ячейку с координатами финиша и будет ответом

  • @TheJayAndrey
    @TheJayAndrey 8 หลายเดือนก่อน

    Для ускорения можно было бы запустить два поиска в ширину одновременно, тогда бы мы могли сократить лишние переборы путей в других направлениях.

  • @ArtM-w9d
    @ArtM-w9d 8 หลายเดือนก่อน +1

    Знаю 2 решения:
    1. За O(1) можно пересчитать по формуле, опирающиюся на полуинвариант сумм
    2. BFS или Дейкстра, оба за O(n) в данном случае

    • @jez_ass2317
      @jez_ass2317 8 หลายเดือนก่อน

      А можно ссылку на объяснение первого решения? Или просто объяснить поподробнее)

  • @jvilbre
    @jvilbre 8 หลายเดือนก่อน

    очень интересная задача. хорошо рассказано. выпускай ролики чаще

  • @rybiizhir
    @rybiizhir 6 หลายเดือนก่อน

    Можно слоями, а можно просто назначить вес каждой клетке.

  • @qrthack
    @qrthack 8 หลายเดือนก่อน +1

    С анимацией хорошо получилось, пожалуйста, делай больше видео

  • @victorarshavskiy4216
    @victorarshavskiy4216 8 หลายเดือนก่อน

    Хорошая задачка, но мне больше интересно как анимация делалась :)

  • @Limitchik
    @Limitchik 8 หลายเดือนก่อน

    1. Полный перебор не нужен - можно оптимизировать:
    1.1. если исходная клетка X

  • @siarheibrazil4110
    @siarheibrazil4110 5 หลายเดือนก่อน

    спс. намешалo бы добавить анализ сложности (временной и по памяти)

  • @panfilovandrey
    @panfilovandrey 8 หลายเดือนก่อน

    Видосы у тебя всегда классыные, и объясняешь хорошо, побольше бы. По самому алгоритму. Слишком ресурсоемко обходить все клетки, особенно для большого поля, например, если выход стоит в 50,50 зачем нам идти в минус? а там будет обход и просчет лишних тысяч вариантов, а если доска 100тыс на 100тыс ? Как-то слишком в лоб, можно оптимизировать, тем более, что мы знаем, что конь всегда сдвигается на 2 и 1 клетку, можно заранее проложить маршрут в направлении выхода, а потом при приближении уже перебирать варианты. Конечно, код будет сложнее, но все зависит от задачи, надо экономить ресурсы или нет.

  • @ЧенЧен-ц1ь
    @ЧенЧен-ц1ь 8 หลายเดือนก่อน

    Считаем разницу по оси X и Y между текущим положением коня и нужной клеткой, получим Sx и Sy. Если Sx и Sy кратны 3, то количество ходов = (Sx + Sy)/3.
    В противном случае if Sx>5 then Sx_first = round((Sx-3)/3, 0); Sx_last = Sx - Sx_first*3 else Sx_first = 0; Sx_last = Sx, то же для Sy, Sу_first, Sy_last.
    Sx_first + Sу_first - количество ходов, что бы оказать в окрестности Sx_last, Sy_last клетки, не более 5 клеток. И вот тут можно применить какой угодно алгоритм.
    В чём преимущество, если доска 1000^1000 на 1000^1000, то со слоями может быть проблема, а так нет.

  • @watchegondu4716
    @watchegondu4716 8 หลายเดือนก่อน +1

    конь на нулевых кордах, мы просто берем число координат, до которых нам нужно дойти и переводим их в положит число, например дали -2 -7, то это считаем как 2 и 7, если -2 7, то 2 7 и тд. берем эти х и у, отнимаем максимальное количество 3 1, 1 3, 6 0 и 0 6 (последняя пара дает sum+=2, первая sum++) до того момента, пока х

    • @jez_ass2317
      @jez_ass2317 8 หลายเดือนก่อน

      Ничего не понятно из твоего объяснения

  • @miyondev09
    @miyondev09 7 หลายเดือนก่อน +1

    un canal ruso que encontré para aprender estructura de datos muy buen canal

  • @sevangulov
    @sevangulov 8 หลายเดือนก่อน

    Задача класс! Больше подобного контента!

  • @fedoslozben
    @fedoslozben 7 หลายเดือนก่อน

    Не,вообще транзитивное замыкание с минимальным весом штука крутая и очень мощная,но тут явно можно докрутить из-за специфики задачи.
    Скажем,вот у нас две точки,которые надо соединить и они образуют как бы прямоугольник и вот выясняется,что если его немного расширить,то любой минимальный путь можно перестроить без потерь так,чтобы он в прямоугольник попал.
    Для удобства перекрутим так всё,чтобы вход был слева снизу и выход справа сверху.
    Понятно,что на ответ это не повлияет,а вот на ход решения повлияет сжатием бреда.
    Докажем это замысловатое (или не особо) утверждение:
    Пускай есть какой-то путь и он неуменьшаем.Тогда все точки,которые выходят за верхнее ограничение мы отразим зеркально вниз,если есть такие две точки,что она на одном уровне.Понятно,что такое преобразование не изменит длинну пути,но применив так много раз потом мы запихаем весь путь в рамки наши.Проблема в том,что они могут быть на разных уровнях,так как по осям у нас переход на 2 или на 1.
    В этом случае придётся воспользоваться оптимальностью пути.
    посмотрим на кусок пути,где вот он начал уходить вверх за границу и в какой-то момент вернулся в в прямоугольник,но на уровень ниже (выше нельзя,так как там кончается граница).Тогда давайте заметим такой факт,что нам важно то именно с точки зрения конца пути чтобы всё было хорошо,а на остальное пофигу.Тогда отразим весь этот подпуть относительно горизонтали,проходящей через конец пути.В случае,если начало проблемного подпути было на одной высоте с точкой выхода проблем не будет,но вот если она была ниже на один,придётся ещё подумать.
    Итак,у нас плохой кусок пути начинается за 1 единицу до конца и заканчивается прямо на конце.Другие случаи мы уже разобрали,этот-последний.
    Тут прям нетривиально!
    Рассмотрим последние два действия.
    У нас проблемы из-за того,что последний ход был на 2 шага по высоте,иначе бы мы нашли что отразить.
    Так же первый шаг не был по высоте на 1,так как иначе после отражения мы посмотрим на 1 шаг до конца и сможем на изи отразить.
    Те первый и последний прыжки были на 2 вверх и вниз.
    Однако раз у нас не совпали высоты,применили прыжок по высоте на 1 нечётное число раз и хотя бы 1 раз.
    А что,если его убрать и вставить как нам надо?
    Тогда мы закончим на 2 клетку по иксу куда-то не туда и на 1 клетки по y куда-то не туда.
    рассмотрим все случаи:(+-2;+1)
    Тут просто,по иксу понятно что на 2 в нужную сторону догонимся и по игреку как раз одной хватит.
    Но что,если (+-2;-1),ведь тогда нам надо уже на 3 двигаться.Надо это как-то проработать.
    Тут просто на самом деле.В каком-то из ходов,где высота по 2 меняем сторону по высоте (как минимум,начало и конец с разными знаками и можно выбрать кандидата (последний после переворота перевернём и кайф)).
    Так же делаем по иксу хитрость нашу,когда слишком вправо.
    Влева и вниз - ну просто думаем,что путь шёл с конца на начало и наши рассуждения все.
    Тут важно,что при любом из описанных действий нехороший кусок теряем как минимум два звена в себе (они становятся хорошими) и конечным количеством таких действий получится всё запихать в коробку.
    Но какая же коробка?
    Как минимум,надо по каждой оси чтобы было длинна коробки 4,иначе не проходит наше приведение.
    Ведь в худшем случае мы сдвигались на 1,при этом замутив крутой перекрут,юзавший ещё 3 ниже.
    А лучшие вообще ничего не делали плохого:самое жёсткое-это на две спускались,чтобы плохой кусок уменишить.
    Те min{4;|start'-finish'|} по каждой оси.
    Ну или,если мы как-то узнали,что в пути нет последнего стрёмного варианта (мало ли),то min{3;|start'-finish'|}.
    Если что,это расстояние от начала и конца до противоположных стенок.Если не хватит,то увеличивать СО ВСЕХ СТОРОН!
    Теперь с чистой совестью можно урезать поле,зная,что все пути будут (хоть и перекрученные) найдены и замыкание именно для последней вершины будет таким же.
    Тут,конечно,важно,что для других вершин это будет не так (хотя,нам какая разница).
    Заметим,что такая оптимизация хоть и сложна в понимании,проста в реализации и при том очень эффективна.

  • @_____55555
    @_____55555 8 หลายเดือนก่อน +2

    Анимация огонь ))
    Но задается вопрос, т.к. мы знаем паттерн движения коня, то почему не исключить несонаправленные вектора для, новых точек и вектора старта и финиша ?

  • @pg4ok611
    @pg4ok611 8 หลายเดือนก่อน +2

    Легенда вернулась, увидимся через год?)

  • @yuryb8310
    @yuryb8310 8 หลายเดือนก่อน

    Задача решается поиском в ширину. Решал аналогичную задачу в универе на экзамене по АСД (Алгоритмы и структуры данных) на третьем курсе

  • @Артёмфомин-ш1ы
    @Артёмфомин-ш1ы 2 หลายเดือนก่อน

    аааа зачем уровни
    массив постоянного размера
    очередь размером в поле
    и ячейка ввиде структуры (visited, from, steps)
    и просто перебираешь каждую ячейку из очереди
    при переборе добавляются новые
    как только достал из очереди новую ячейку то проделываешь с ней все действия проверяя в начале на visited и в конце проверяешь, если это target, тогда пишешь ответ, но продолжаешь досчитывать, чтобы в последствии можно было использовать данный массив для нахождения количества шагов до любой ячейки от того же самого положения коня за O(1), просто ткнув в этот массив пальцем, а там уже будет ответ в поле steps у ячейки
    а если случится такое что у тебя в очереди закончились ячейки то говоришь что пути нет
    в итоге работает даже для не ровного поля и с островами
    а очередь размером в поле потому что возможно построить поле где в стеке будет лежать больше половины ячеек

  • @opalev
    @opalev 8 หลายเดือนก่อน +6

    Спасибо!
    1. хотелось бы в коде видеть номера строк, чтобы ссылаться на них в комментах
    2. не увидел ограничения доски, возможно мне стоит самому реализовать, но кажется, находясь в x=2 y=1 мы выскочим за пределы доски.
    3. не увидел где инициализируются x, y, возможно это особенность, или просто опустили из примера.
    4. что если выход будет в [-1, -1] ?

    • @kotiko_oks
      @kotiko_oks 8 หลายเดือนก่อน

      про доску в начале объяснялось, про то, что она бесконечная не сказано, но обрати внимание на координаты. Если [1,1] Это стартовая позиция

    • @opalev
      @opalev 8 หลายเดือนก่อน

      @@kotiko_oksвидимо нужно самому реализовать с тестами)

    • @widny31
      @widny31 8 หลายเดือนก่อน

      2. Ограничение не нужно в контексте задачи. Ограничения доски представлены максимальным и минимальным значением типа int в python
      3. (x, y) - это итератор в цикле for, который там же инициализируется
      "for (x, y) in layer" перебирает все пары (кортежи) (x, y), которые находятся в layer
      4. Даже чисто математически не имеет значения, если выход находится на отрицательных координатах

    • @opalev
      @opalev 8 หลายเดือนก่อน

      @@widny31 2-3 понял, питон не знаю просто. 4 суть не в отрицательных координатах, а в том, что ограничения доски я не видел, поэтому решил, что конь просто уйдёт за пределы и никогда не вернётся в -1,-1.

  • @eduardtsuranov712
    @eduardtsuranov712 8 หลายเดือนก่อน

    Это же по сути полный перебор, а значит алгоритм не эффективный.
    Можно изучить свойства движения коня и вывести некую формулу. Типа идем на прямую к целевой клетке, пока она не попадет в заранее просчитанный квадрат 5*5. Возможно, можно просто формулу вывести четкую

  • @АнатолийПетрунин-ц1ч
    @АнатолийПетрунин-ц1ч 8 หลายเดือนก่อน

    когда лет 20-25 назад делал игру Lines на дельфи аналогично просчитывал путь для шарика из исходной точки в точку в которую кликнул игрок

  • @Mikjagger1
    @Mikjagger1 8 หลายเดือนก่อน

    Альманах...круто, мне нравится

  • @zhdanvadim9536
    @zhdanvadim9536 8 หลายเดือนก่อน

    Спасибо, хорошая задача и разбор

  • @KogutIvanTutoring
    @KogutIvanTutoring 8 หลายเดือนก่อน

    А теперь попробуйте решить ту же задачу, но слегка сложней:
    Есть вместо одного k коней на поле. Нужно для каждой клетки поля понять за какое минимальное количество ходов один из этих коней доберется до нее

  • @orange-vlcybpd2
    @orange-vlcybpd2 8 หลายเดือนก่อน +2

    Какие 7.5к € джуниору в Германии?; 90к в год даже техлиды не все получают. В 4к брутто - поверю, если данные есть: законченый универ, например, с дуальным обучением.

  • @alexanderspeshilov839
    @alexanderspeshilov839 8 หลายเดือนก่อน +1

    Хм. Зачем там хеш-таблица, если проще и быстрее использовать, например, очередь? Я в обходе в ширину обычно просто добавляю в конец очереди новые вершины-кандидаты на обход. Вершины (в данной задаче клетки), которые уже обошли, хранить в отдельной структуре (в данной задаче я бы использовал двумерный массив, но можно и хеш-таблицу).

  • @rocknrolla1440
    @rocknrolla1440 8 หลายเดือนก่อน

    Добрый день, Александр, отличное видео! Вы устраивались на Junior Software Engineer, я правильно понимаю?

  • @ДиванныйКритик-к7л
    @ДиванныйКритик-к7л 4 หลายเดือนก่อน

    спасибо

  • @DmitryShubin-ym4pj
    @DmitryShubin-ym4pj 8 หลายเดือนก่อน

    Хорошее решение. Мысленно представил как решать это на C++, но на Питоне, конечно, компактнее получается решение. Единственный неявный для меня момент алгоритма - это то, что запоминается только предыдущий и текущий слои. Видимо, подразумевается, что с более старыми слоями пересечения быть не может. Приму на веру, видимо, есть строгое доказательство данного факта.

    • @Abingusus
      @Abingusus 8 หลายเดือนก่อน

      Если граф двунаправленный, то предыдущий слой это заслонка для всех возможных путей обхода. Можешь представить длинный коридор с дверьми, где ты захлопываешь за собой дверь, но предпредыдущая всегда открывается после этого. Но до нее ты не дойдешь, т.к. уже захлопнул дверь, которая ведет к ней.

    • @DmitryShubin-ym4pj
      @DmitryShubin-ym4pj 8 หลายเดือนก่อน

      ⁠@@Abingususчто такое «двунаправленный граф»? Я знаю только неориентированные и ориентированные графы. Я не понял, как ваш комментарий относится к тому, что я написал выше. В теории конь не обязан не возвращаться к клеткам из более старых слоев.

  • @АлексейРакитин-э9ю
    @АлексейРакитин-э9ю 8 หลายเดือนก่อน

    круто, спасибо за видосы!

  • @ii3246
    @ii3246 8 หลายเดือนก่อน

    решение не лучшее, но сойдет. особенно для джуна.😁
    ну шахматная доска это матрица. это раз.
    врытое уже зависит что именно надо! просчитать ходы - это граф. узнать где какая фигура - это матрица.
    третье, структуры данных все же стоит подтянуть.😁

  • @n3nikita
    @n3nikita 8 หลายเดือนก่อน

    через 3 недели у меня тоже собеседование в амазон. в алгоритмах я слаб и учу их практически с нуля.
    прежде чем посмотреть видео, поставил его на паузу и попробовал решить задачку сам. получилось решить ее за 5 минут. попадется ли мне подобная задача на собеседовании, это уже как повезет.

  • @jekahamster8041
    @jekahamster8041 8 หลายเดือนก่อน +1

    Вижу недостаток в этом алгоритме. Тут все таки есть повторные посещенния уже пройденых клеток, так как сохраняются лишь два послдних уровня дерева (слоя). Нам не нужно повторно добавлять клетки, в которых мы уже были, потому как для них уже уже известно минимальное число шагов, а так же они породят огромные поддеревья. Что бы избежать этой проблемы, можно добавить либо множество всех посещенных вершин, либо битсет

  • @Holstinin007
    @Holstinin007 8 หลายเดือนก่อน

    Лучший мотиватор!

  • @cd6693
    @cd6693 8 หลายเดือนก่อน

    спасибо за видео

  • @alex_pro_fpv
    @alex_pro_fpv 8 หลายเดือนก่อน

    Спасибо за задачу

  • @rasimbot
    @rasimbot 8 หลายเดือนก่อน +1

    Неужели там не появляется паттерн, чтобы можно было написать формулы, в которые можно подставить исходную и целевую координаты и сразу получить ответ, по крайне мере для ситуаций, когда исходная и целевые поля не у края доски?

    • @Abingusus
      @Abingusus 8 หลายเดือนก่อน

      Можно за линейную, написал коммень, но кажется можно за консьантную сложность

  • @Game_Pro_
    @Game_Pro_ 8 หลายเดือนก่อน +2

    Стоит ли идти учиться на программиста если уже одну профессию закончил но она мне не нравится? Мне уже 23 года - стоит ли рисковать и переучиваться на программиста?

    • @u_n_d_e_r_s_c_o_r_e_d
      @u_n_d_e_r_s_c_o_r_e_d 7 หลายเดือนก่อน

      главное чтоб нравилась профессия

  • @PimiTree
    @PimiTree 8 หลายเดือนก่อน

    Я бы решал через вектор до выхода - т.е. какой из возможных ходов сокращаяет дистанцию до выхода

  • @Seniorius
    @Seniorius 8 หลายเดือนก่อน

    Спасибо

  • @ЕвгенийП-д8л
    @ЕвгенийП-д8л 2 หลายเดือนก่อน

    Очень лëгкая задача. Концепт решения верный, а прога не работает. Нет проверки на выход за границы доски, предыдущий слой не нужен, нужно шаги до всех ячеек хранить и лучше это делать в матрице. Ячейка может оказаться недостижимой и при такой реализации прога зациклится.

  • @DiabloZet_
    @DiabloZet_ 8 หลายเดือนก่อน

    Слишком ресурсоемкие затраты, не легче для начала отрезать длину перемещения коня +3 за шаг от позиции к близлежайшей точке где будет прямолинейное перемещение к цели, а там уже рассчитать в конце более краткий путь, ибо затраты на расчёт к каждой точке если это доска на 100млн, то расчёт всех слоев будет очень проблемным.

  • @igorf243
    @igorf243 8 หลายเดือนก่อน +1

    осталось только посмотреть что такое A* и ускорить нахождение решения в среднем в 4 раза

    • @ReAgent003
      @ReAgent003 8 หลายเดือนก่อน

      агонь!

  • @iklintsov
    @iklintsov 8 หลายเดือนก่อน

    Обычный, но самый не эфективный алгоритм. Кто может сказать О если точни начало и конец находятся далеко друг от друга. Полиномия по любому. А решение проще. двигаемся на прямую в сторону исходной точки максимально быстро, останавливаясь на расстоянии одного прыжка от нее. там мы либо совпадем, либо придется прыгнуть в сторону 1 или 2 раза. Как быстрее прыгать вычисляем по формуле что больше |х1 - х2| или |y1 - y2|. Дальше смотрим, что делится на 2? Соответственно прыжки можно просто посчитать. Немного на пивасике, могу не очень ясно объяснять.

    • @ReAgent003
      @ReAgent003 8 หลายเดือนก่อน +2

      я тебя ясно понял потому что сам на пивасике

    • @yd8874
      @yd8874 8 หลายเดือนก่อน +1

      По этому алгоритму с расстояния 3,4 надо прыгать в 2,2. Но это неправильно, потому что из 2,2 придётся 4 хода добираться. Надо прыгать в 1,3, откуда 2 хода.
      Мне кажется, что надо захардкодить таблицу решений x,y от 0 до 4, а в неё по вашему алгоритму добираться.

  • @RedBallOfLove
    @RedBallOfLove 8 หลายเดือนก่อน +1

    есть подозрение, что задача решается проще, но мне лень сейчас думать :)

  • @MRaynold
    @MRaynold 8 หลายเดือนก่อน

    Спасибо!

  • @VasyaVasin-ep8sn
    @VasyaVasin-ep8sn 8 หลายเดือนก่อน

    Что мне нужно делать, что бы понимать о чем ты вообще говоришь)) Я вообще не улавливаю, что нужно изучать в самом начале ? Я хочу знать, но полагаю нужно дойти до этого, а что бы дойти, нужно с чего то начать, скажи с чего начать, если я даже не понимаю о чем ты говоришь :DDD

  • @davidola1814
    @davidola1814 6 หลายเดือนก่อน

    Это вроде простой перебор всех возможных вариантов

  • @Rice0987
    @Rice0987 8 หลายเดือนก่อน

    Так, если координаты конечной точки известны, то может можно искать только в направлении заданой четверти, чтобы не перебирать 3/4 заведомо не нужных направлений?

  • @pavelpavel7938
    @pavelpavel7938 7 หลายเดือนก่อน

    Почему из первой точки 8 ходов, если всего 2, а остальные являются их отражением? Сильно подозреваю, что достаточно просто создать матрицу n*n, для одного квадранта, а потом вычислить координаты точки в этих матрицах.

  • @kircatle
    @kircatle 7 หลายเดือนก่อน

    Здравствуйте, Александр, а Лукин Владимир Николаевич не ваш родственник (в маи преподаёт) ?

  • @Ares8243-d5w
    @Ares8243-d5w 8 หลายเดือนก่อน +1

    Единственное в чем не вижу смысла - проверка на наличие элемента в сете, т к сет нем может содержать повторяющихся элементов. Он автоматически их убирает

    • @Ares8243-d5w
      @Ares8243-d5w 8 หลายเดือนก่อน

      Возможно, я где-то ошибаюсь. В любом случае спасибо за ролик, анимации идеально подходят для понимания.

    • @Ares8243-d5w
      @Ares8243-d5w 8 หลายเดือนก่อน

      Я опирался на то, что код написан на Python, но, видимо, это не так. В любом случае, можешь скинуть свой код в текстовом виде?

  • @vasinas81
    @vasinas81 8 หลายเดือนก่อน

    Можно добавить эвристическую функцию и получить AStar (будет быстрее)

  • @sasasupov8597
    @sasasupov8597 8 หลายเดือนก่อน

    Я так понимаю, доска задачи бесконечная? Функция moves не смотрит на границы доски.

  • @igornovomirskiy8216
    @igornovomirskiy8216 8 หลายเดือนก่อน +2

    Да ну нах..лучше в таксисты пойду.

  • @freehck
    @freehck 8 หลายเดือนก่อน +1

    Угу, вот смотрю на эти красивые превью вида "7500 евро в месяц", и понимаю, что ничего не изменилось в США/ЕС. Как не платили, так и не платят. =)
    PS: для тех, кто не понял, о чём речь -- у них принято зп мерить без налогов и социалки, можете смело вычитать до половины; потом вычитаете стоимость жизни и удивляетесь, как мало остаётся; притом 7-9к это при релокации, а если вы удалёнщик -- рейты сразу падают до 4-5к, и это норма. Но что забавно, чисто по финансам 5к на удалёнке выгоднее, чем 9к с релоком.

  • @zheka77111
    @zheka77111 7 หลายเดือนก่อน

    Четко

  • @alekseypavlov2539
    @alekseypavlov2539 3 หลายเดือนก่อน

    Решение на JS, такое-же как в видео
    function findMinSteps(END_POINT = [ 1, 1 ], START_POINT = [ 5, 5 ]) {
    let step = 0;
    let combinations = 0;
    let currentLayer = new Set([ START_POINT.toString() ]);
    let prevLayer = new Set();
    while (!isFindDistance(currentLayer)) {
    let nextLayer = new Set();
    currentLayer.forEach((point) => {
    const _point = point.split(",");
    moves(+_point[0], +_point[1]).forEach((newPoint) => {
    if (!prevLayer.has(newPoint)) {
    nextLayer.add(newPoint);
    }
    });
    });
    [ prevLayer, currentLayer ] = [ currentLayer, nextLayer ];
    step++;
    }
    function isFindDistance(layer) {
    combinations += layer.size;
    return layer.has(`${ END_POINT[0] },${ END_POINT[1] }`);
    }
    function moves(x, y) {
    const points = new Set();
    points.add([ x + 1, y + 2 ].toString());
    points.add([ x - 1, y + 2 ].toString());
    points.add([ x + 2, y + 1 ].toString());
    points.add([ x + 2, y - 1 ].toString());
    points.add([ x + 1, y - 2 ].toString());
    points.add([ x - 1, y - 2 ].toString());
    points.add([ x - 2, y + 1 ].toString());
    points.add([ x - 2, y - 1 ].toString());
    return points;
    }
    return { step, combinations };
    }
    const now = performance.now()
    console.log(findMinSteps([ 227, 369 ]));
    console.log((performance.now() - now) / 1000)

  • @nobodyishere8935
    @nobodyishere8935 8 หลายเดือนก่อน

    Can we see the same BFS done recursively ? Would it be an easier way to implement if you have just 15 minutes?

  • @slmpnv
    @slmpnv 8 หลายเดือนก่อน

    Мне в голову dict пришёл. Можно ли это реализовать, если layer_number - ключ, а сет таплов(или списков) с доступными координатами - значение?