Task from JS interview: Count the number of islands in the matrix | Number of Islands

แชร์
ฝัง
  • เผยแพร่เมื่อ 22 ม.ค. 2025

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

  • @nnnabbot
    @nnnabbot ปีที่แล้ว

    Мне нравится, что все очень понятно и темы интересные . Мне было интересно как подобные обходы строятся. Рекурсия в дополнительной функции очень круто

  • @AShahabov
    @AShahabov 3 ปีที่แล้ว +8

    Бомба, а не видео!!! Спасибо большое. Наконец-то дошло до меня как работает этот алгоритм.

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว

      Рад что было полезно!

  • @igorkirey1642
    @igorkirey1642 3 ปีที่แล้ว +3

    Сам не смог решить, всё пытался какие-то схемы придумать, а оказалось так просто. Спасибо за разбор!

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว

      Рад что было полезно!

  • @yuryitikhonoff9631
    @yuryitikhonoff9631 3 ปีที่แล้ว +6

    Юухуууу. Вот и разбор новой задачки!!! Респект!

  • @koshakkoshakov7104
    @koshakkoshakov7104 3 ปีที่แล้ว +7

    Спасибо за контент с задачами! Очень интересно и познавательно!

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว +3

      Рад, что было полезно! Благодарю за поддержку!

  • @SerzhNesteruk
    @SerzhNesteruk ปีที่แล้ว +1

    Спасибо за интересную задачу!
    В собственном решении я немного заморочился с иммутабельностью, поэтому преобразовал grid в матрицу объектов. Обход всех вершин графа (ячеек острова) тоже реализовал через рекурсию, но для соблюдения принципа DRY добавил абстракцию массива векторов. Попадание координат в границы матрицы проверял выражением: map[y]?.hasOwnProperty(x), что эквивалентно двойному условию: map[y] && x in map[y].
    Вот, что получилось:
    const numIslands = grid => {
    const map = Array.from(Array(grid.length), (_, y) => grid[y].map((v, x) => ({
    x: x,
    y: y,
    visitable: v === '1',
    })));
    function traversal(entrance, map) {
    entrance.visitable = false;
    const directions = [
    {x: 1, y: 0}, // right
    {x: 0, y: 1}, // down
    {x: -1, y: 0}, // left
    {x: 0, y: -1}, // up
    ];
    for (const vector of directions) {
    const x = entrance.x + vector.x;
    const y = entrance.y + vector.y;
    if (map[y]?.hasOwnProperty(x) && map[y][x].visitable) {
    traversal(map[y][x], map);
    }
    }
    }
    let counter = 0;
    for (const row of map) {
    for (const cell of row) {
    if (cell.visitable) {
    counter++;
    traversal(cell, map);
    }
    }
    }
    return counter;
    };

  • @ІльченкоАртем
    @ІльченкоАртем 3 ปีที่แล้ว

    Счастлив что нашел Ваш канал!! Js react node задачи собеседование IT full stack frontend backend

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว

      Супер! Рад это слышать! Очень вдохновляют такие комментарии! Успехов, друг!

  • @DmitryDolganov
    @DmitryDolganov 3 ปีที่แล้ว +4

    Благодарю!

  • @EvilYou
    @EvilYou ปีที่แล้ว +1

    Довольно простая задача, особенно если можно изменять исходный массив:
    function numIslands(grid) {
    let counter = 0;
    for (let i = 0; i < grid.length; i++) {
    for (let j = 0; j < grid[0].length; j++) {
    if (grid[i][j] === "1") {
    paint(i, j);
    counter++;
    }
    }
    }
    function paint(r, c) {
    grid[r][c] = "2";
    if (grid[r - 1]?.[c] === "1") paint(r - 1, c);
    if (grid[r][c + 1] === "1") paint(r, c + 1);
    if (grid[r + 1]?.[c] === "1") paint(r + 1, c);
    if (grid[r][c - 1] === "1") paint(r, c - 1);
    }
    return counter;
    }

  • @multiply87
    @multiply87 3 ปีที่แล้ว

    офигеть, как круто) 2 часа потратил на алгоритмы, но так и не придумал, как это правильно реализовать. Пробовал сбор индексов непустых элементов, а потом их сравнение, но вылезли нюансы. Оказалось нужно поменять исходный массив и применить рекурсию) Спасибо за опыт, это реально пригодится)

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว +3

      Классно что сам вначале пытался решить! Восхищаюсь твоим упорством!
      Менять исходный массив тут прямо обязательности нету - можно и копию сделать. Но тогда сложность по памяти вырастет.

    • @multiply87
      @multiply87 3 ปีที่แล้ว

      @@frontendscience проблемы оптимизации для меня актуальны. Потому очень рад, что нашел твой канал. Спасибо тебе за труд. Хорошие алгоритмы всегда пригодятся)

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว +1

      А Вы более разумного применения своему времени, чем писать 100500 комментов о том, что Вы не поняли решения? Например, чтоб разобраться, почему там таки должна быть рекурсия.

    • @multiply87
      @multiply87 3 ปีที่แล้ว

      @@frontendscience ну куда ж без хейтеров? :)

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว

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

  • @radist126
    @radist126 ปีที่แล้ว

    Афигенно объяснение!!!❤ Спасибо

  • @hotremz
    @hotremz 3 ปีที่แล้ว +3

    Круто: доступно и понятно.

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว +1

      Рад, что было полезно

  • @sony1939
    @sony1939 2 ปีที่แล้ว

    Спасибо друг! Помог решить задачу!

  • @nnnabbot
    @nnnabbot ปีที่แล้ว

    Спасибо большое

  • @denmccormik7654
    @denmccormik7654 3 ปีที่แล้ว +4

    Сергей, надеюсь увидете этот комментарий. У меня на собесе была задачка, до сих пор не могу придумать как ее решить - может быть разберете в следующих видео?
    У нас есть матрица неизвестного размера - это фото местности где каждое число в матрице высота над уровнем моря. Нужно найти самый эфективный путь марсоходу из верхней левой точки в правую нижнюю. Каждый шаг стоит 1 единицу топлива. Также разница между высотами это топливо на спуск или подъем. Бывают недостижимые точки они обозначаются как Х. А САМОЕ СЛОЖНОЕ - первый шаг по диагонали стоит как обычный - единицу, второй в два раза больше, третий в два раза и так далее.
    Без диагональных премудростей Дейкстрой все считается, все супер. Главное граф из матрицы сделать и все гуд.
    Но вот диагонали - не пойму как с рими разобраться.

    • @denmccormik7654
      @denmccormik7654 3 ปีที่แล้ว

      Пример
      01
      25
      Из нуля в двойку - 3 единицы топлива = 1 + разница высот два минус ноль

    • @danoxztm3250
      @danoxztm3250 3 ปีที่แล้ว +1

      Капец задачка, ты там не в NASA устраиваешся, чтобы марсоходами управлять?

    • @thomasgabe3588
      @thomasgabe3588 3 ปีที่แล้ว

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

    • @denmccormik7654
      @denmccormik7654 3 ปีที่แล้ว

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

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว

      Привет! Это на какую должность тебе такую задачу задали 🤯?
      Из идей которые есть: попробуй представить диагональ как ещё одно ребро в графе. То-есть выйдет плюс несколько рёбер везде.

  • @benya4396
    @benya4396 3 ปีที่แล้ว +1

    Не могу понять почему мы в случае колонок не делаем проверку и при этом не вываливаемся за границы массива?

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว

      У нас есть проверка :) внутри if если мы обращаемся к несуществующей ячейке то получаем undefined. А он внутри if приводится к false

  • @GreenGotty
    @GreenGotty 3 ปีที่แล้ว

    В голову пришло только топить острова клетка за клеткой.. не оптимально, конечно, но прокатило! :)
    Runtime: 92 ms, faster than 56.10% of JavaScript online submissions for Number of Islands.
    Memory Usage: 42.7 MB, less than 34.10% of JavaScript online submissions for Number of Islands.
    var numIslands = function (grid) {
    const lenY = grid.length;
    const lenX = grid[0].length;
    var total = 0;
    var y = 0;
    var x = 0;
    while (y < lenY) {
    if (grid[y][x] == '1') {
    collapseIsland(y, x);
    total++;
    nextTile(2);
    } else {
    nextTile(1);
    }
    }
    return total;
    function collapseIsland(yy, xx) {
    grid[yy][xx] = '0';
    if (grid[yy + 1] && grid[yy + 1][xx] == '1') {
    grid[yy + 1][xx] = '0';
    collapseIsland(yy + 1, xx);
    }
    if (grid[yy][xx + 1] && grid[yy][xx + 1] == '1') {
    grid[yy][xx + 1] = '0';
    collapseIsland(yy, xx + 1);
    }
    if (grid[yy][xx - 1] && grid[yy][xx - 1] == '1') {
    grid[yy][xx - 1] = '0';
    collapseIsland(yy, xx - 1);
    }
    if (grid[yy - 1] && grid[yy - 1][xx] == '1') {
    grid[yy - 1][xx] = '0';
    collapseIsland(yy - 1, xx);
    }
    }
    function nextTile(step) {
    if (x >= lenX - step) {
    x = 0;
    y++;
    } else {
    x += step;
    }
    }
    };

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว +1

      Благодарю за решение!
      "Все острова успешно затоплены!" :)

    • @GreenGotty
      @GreenGotty 3 ปีที่แล้ว

      😁🏴‍☠️

  • @umarjuraev1624
    @umarjuraev1624 3 ปีที่แล้ว

    Спасибо большое за контент

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว

      Рад, что понравилось!

  • @evgeniychip
    @evgeniychip 3 ปีที่แล้ว +1

    здорово!)

  • @sb-dor
    @sb-dor 3 ปีที่แล้ว +3

    Сразу лайк

  • @АртемШкитко-ц9г
    @АртемШкитко-ц9г 3 ปีที่แล้ว +2

    а проверку на существование елемента массива слева и справа не нужно(как в случае с сверху и снизу)?

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว +1

      Она есть ) даже ячейки специально в анимации подсвечены когда это делали.

    • @АртемШкитко-ц9г
      @АртемШкитко-ц9г 3 ปีที่แล้ว

      ​@@frontendscience да не, я про grid.?[R].?[C-1]==='1' - вот так делать не нужно? (.?)

    • @АртемШкитко-ц9г
      @АртемШкитко-ц9г 3 ปีที่แล้ว +1

      @@frontendscience мы же попадаем сразу на первый элемент первой строки, и относительно этого элемента, слева нет ничего, и если будет массив [0,0,0,0,0,1] - то тут справа ничего не будет.

    • @АртемШкитко-ц9г
      @АртемШкитко-ц9г 3 ปีที่แล้ว

      разобрался) избыточное условие будет, можно и не писать

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว +2

      А я теперь понял изначальный вопрос )

  • @tatianovnafrutti8982
    @tatianovnafrutti8982 3 ปีที่แล้ว +2

    Жесть.. сложная фигня какая-та)) Посмотрев , пришла мысль бросить занятие фронтом.. Бо если мне бы такое попросили сделать, у меня бы 100% не вышло((( мде.. вот те и войти в ит(( . Спасибо за классный видос

    • @Сергей-у6и7б
      @Сергей-у6и7б 3 ปีที่แล้ว +2

      Да тебе лишь бы руки опустить. Данная задача на уровень middle/senior (и то в каком-нибудь яндексе), но точно не на начинающего или джуниора.

    • @tatianovnafrutti8982
      @tatianovnafrutti8982 3 ปีที่แล้ว +1

      @@Сергей-у6и7б А ну если не на начинающего то норм.

    • @МаксимСоловьев-с9н
      @МаксимСоловьев-с9н ปีที่แล้ว

      не бросили фронт по итогу?)

    • @tatianovnafrutti8982
      @tatianovnafrutti8982 ปีที่แล้ว

      @@МаксимСоловьев-с9н Нет.. Работу найти сейчас не могу но уже есть опыт и на фрилансе по немногу делаю по мелочи

  • @kirillkrutsko8399
    @kirillkrutsko8399 10 หลายเดือนก่อน

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

  • @qrthack
    @qrthack 3 ปีที่แล้ว +1

    можно узнать, где вы делаете картинки к видео, как на 7:40 правый нижний угол?

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว

      excalidraw.com

    • @qrthack
      @qrthack 3 ปีที่แล้ว

      @@frontendscience спасибо большое

  • @crecker05ru
    @crecker05ru 3 ปีที่แล้ว

    2:42 а если один массив будет больше чем grid[0].length ?что тогда?

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว

      В условии нашей задачи этого быть не может - у нас прямоугольная "карта"

    • @crecker05ru
      @crecker05ru 3 ปีที่แล้ว

      @@frontendscience ок :)

  • @alexanderalexandrov3972
    @alexanderalexandrov3972 3 ปีที่แล้ว

    Невнимательно прочитал задание. ПОдумал, что это питон. Подумал сколько всего интересного я еще не знаю. Увидел JS - Выдохнул... =)

  • @Helen-do2og
    @Helen-do2og 3 ปีที่แล้ว +2

    Классное видео, спасибо! Есть вопрос, сколько по времени нормально будет решать подобную задачку для мидла? Или уровнем easy. Спрашиваю потому, что недавно решала задачи на собеседовании и каждая заняла от 10 до 15 минут - и не понимаю, это было нормально или надо быстрее соображать))

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว +4

      Для медиум задачи 10-15 мин ок, для изи уровня стараться 5-10 мин (они тоже бывают разные по сложности). Если решать много задач на подготовке, то очень быстро на собеседовании сможешь определиться с самим алгоритмом и тогда времени на решение будет уходить в разы меньше.

  • @noo1xp
    @noo1xp 3 ปีที่แล้ว +1

    В 38 строке надо добавить optional changing либо перенести эту строку под первый if, иначе будет ошибка в случае с пустой матрицей

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว

      Да верно. optional changing туда стоит добавить. С другой стороны, на leetcode по условию не может быть пустого массива. Так что 39ю строку вообще можно выкинуть.

  • @ihor-pidhornyi
    @ihor-pidhornyi 3 ปีที่แล้ว +1

    Классное решение, но я бы добавил создание копии массива)

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว

      Это увеличит сложность алгоритма по памяти, что идет в разрез с условиями задачи.

    • @ihor-pidhornyi
      @ihor-pidhornyi 3 ปีที่แล้ว

      @@frontendscience В целом как задачи для алгоритма, я с вами частично согласен, но зайдя на leetcode и проверив 2 кейса, с копированием и без, могу сказать, что разница в 1.4% по памяти не нарушила ничего в условии задачи)

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว +3

      @@ihor-pidhornyi то что на литкод сами тест кейсы не всегда «большие», известный факт. Вопрос в том что тут не нужно делать копию массива. Это сендбокс, это конкретная задача, и копирование тут излишне. ПС: в продашен коде никто конечно мутировать массив не будет!

    • @ihor-pidhornyi
      @ihor-pidhornyi 3 ปีที่แล้ว +1

      @@frontendscience Воот, выражения про прод мне не хватало😅😅😅

    • @SerzhNesteruk
      @SerzhNesteruk ปีที่แล้ว

      ​@@frontendscience Иммутабельность объектов переданных в функцию всё таки считается хорошей практикой (даже для решения задачек). Незначительное увеличение сложности алгоритма по памяти - не критично. А при больших тест-кейсах, где габариты острова будут исчислятся десятками тысяч, у нас может возникникать совершенно другая проблема: переполнение стека вызовов рекурсивной функции.

  • @ВениаминТрепачко
    @ВениаминТрепачко 3 ปีที่แล้ว

    Задача классная, но я не увидел необходимости помечать шестерками другие единицы. Мое решение заключается в том, чтобы проверять предыдущие числа по горизонтали и по вертикали. То есть, если grid[R][C] === "1" и grid[R - 1][C] === "0" и grid[R][C - 1] === "0", это означает что наша единица - начало нового острова. И тогда мы увеличиваем counter. В противном случае (если перед ней или над ней тоже единица) она является продолжением уже посчитанного острова.
    Вот так это выглядит:
    const countIslands = (matrix) => {
    let total = 0;
    if (matrix.length === 0) return 0;
    for (let row = 0; row < matrix.length; row += 1) {
    for (let index = 0; index < matrix[row].length; index += 1) {
    const current = matrix[row][index];
    if (current === 1) {
    const prevRowValue = matrix[row - 1] ? matrix[row - 1][index] : 0;
    const prevIndexValue = matrix[row][index - 1] ? matrix[row][index - 1] : 0;
    if (prevRowValue === 0 && prevIndexValue === 0) {
    total += 1;
    }
    }
    }
    }
    return total;
    };

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว

      Дело в том что острова могут быть не только прямоугольными. Например он может быть «змейкой».

    • @ВениаминТрепачко
      @ВениаминТрепачко 3 ปีที่แล้ว

      @@frontendscience Да. Но в любом случае, если единица является продолжением острова, то либо слева, либо сверху тоже будет единица. Даже в случае со змейкой. Именно такие случаи я и проверяю, так что мое решение отработает корректно.

    • @smashno
      @smashno 3 ปีที่แล้ว

      @@ВениаминТрепачко К сожалению твой код выдает не верный ответ даже на простейших случаях: input [["1","1","1","1","0"]] output: 0, Expected: 1

    • @ВениаминТрепачко
      @ВениаминТрепачко 3 ปีที่แล้ว

      @@smashno Тут проблема исключительно в том, что вы используете массив строк, а я массив чисел 🙂Но логически ведь всё верно

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว

      @@ВениаминТрепачко Логически тоже не верно (если преобразовать строки в числа):
      Input [["1","1","1"],["0","1","0"],["1","1","1"]]
      Output: 2
      Expected: 1

  • @АртемЧепурняк-ж5ъ
    @АртемЧепурняк-ж5ъ 3 ปีที่แล้ว

    В строчках 43 и 44 нужно было так же сделать как и в 45 и 46, потому что если мы проверяем 0вой элемент в массиве то левый сосед будет последний элемент, который может быть отдельным островом, для последнего элемента так же. Или я ошибаюсь?

    • @АртемЧепурняк-ж5ъ
      @АртемЧепурняк-ж5ъ 3 ปีที่แล้ว

      А нет, в js невозможно как в питоне получать доступ типа arr[-1], возвращает undefined

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว +1

      если ты обращаешься к элементу "за границей" массива arr[length+1] arr[-1] то всегда будет undefined

  • @jorgenUA
    @jorgenUA 3 ปีที่แล้ว

    относительно сложности: цикл в цикле же - квадратичная сложность. Нет? С тебя видео про это в любом случае!

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว

      у нас один цикл проходит по всем строкам, а второй по всем столбцам вот и выходит O(n*m). Квадратичная сложность получается когда у нас оба цикла проходятся по всем элементам.

    • @jorgenUA
      @jorgenUA 3 ปีที่แล้ว

      @@frontendscience но количество операций будет зависеть от размера массива как n * n, т. е. n2? Проход по столбцу подразумевает же перебор элементов столбца? Интересно узнать.

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว +1

      n*m - это количество всех элементов в нашей матрице. ты просто итерируемся по всем элементам матрицы. В худшем случае у нас например когда все элементы матрицы единицы. тогда на первой же итерации наших циклов мы зайдем на остров(первый элемент с 1) и еще раз пройдем все наши элементы массива в рекурсии - но сделаем это только один раз, на всех других итерациях цикла у нас уже будут помечены ячейки 6ками - и мы в рекурсивную функцию заходить не будем. так что результирующая сложность останется O(n*m)

  • @sergeyv.8038
    @sergeyv.8038 3 ปีที่แล้ว +1

    Перебор массива осуществляется не из рандомной позиции,
    поэтому в следующих проверках markNeighbours нет необходимости:
    if (binaryMatrix?.[R-1]?.[C] === '1') { markNeighbours(binaryMatrix, R-1,C)}
    if (binaryMatrix[R][C-1] === '1') { markNeighbours(binaryMatrix, R,C-1)}
    Ко всему код можно было бы оптимизировать.
    Сейчас он проходится по тем элементам, которые были помечены "6"

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว +6

      Убрать эти строки нельзя! В тех простейших случаях что я показал в видео, да это бы сработало. Но вот 2 тест кейса когда без этих строк будет не верный ответ:
      [["1","1","1"],["0","1","0"],["1","1","1"]] //Output 2 Expected 1
      [["1","0","1","1","1"],["1","0","1","0","1"],["1","1","1","0","1"]] //Output 2 Expected 1
      По поводу прохода уже посещенных ячеек - эта оптимизация не даст улучшения общей сложности O(n*m). Но вот код значительно усложнит - поэтому не рекомендую оверинжинирить на собеседованиях с такими оптимизациями.

    • @sergeyv.8038
      @sergeyv.8038 3 ปีที่แล้ว

      @@frontendscience Ok. Не рассматривал такой кейс.

    • @de_zavtraa
      @de_zavtraa 3 ปีที่แล้ว +1

      @@frontendscience тоже хотел написать насчёт этого. Спасиба за такие примеры, сходу не самые очевидные

  • @dimitriskidis
    @dimitriskidis 3 ปีที่แล้ว

    Спасибо за видео! Есть вопрос, пытался прогнать код в VS, в консоли в браузере выдает Unexpected token '.', ругаясь на 43 строку binaryMatrix?.[R-1]?.[C] === '1'
    UPD: гуглил, понял, что это за оператор, но при всем, пока не понимаю, как устранить ошибку

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว +1

      Скорее всего в vscode выбрана для запуска старая версия ноды. Я не спец по vscode так что не подскажу как там поставить посвежее чтобы поддерживал этот оператор. Если что в последних версиях хрома все отлично работает. Можно там запустить

    • @dimitriskidis
      @dimitriskidis 3 ปีที่แล้ว

      @@frontendscience понял, спасибо!!

  • @supercereal4667
    @supercereal4667 3 ปีที่แล้ว

    Мне кажется можно добиться более оптимального решения.
    Я прохожу по каждой ячейке матрицы, если она единица, то if (верхняя и левая ячейки === 0 или === undefined), то мы считаем остров. Если условие нарушается, то этот остров уже был посчитан.
    Это решение без доп.рекурсивной функции, поэтому мы не столкнемся с возможными проблемами переполнения стека вызова функ.

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว +3

      Вы уверены, что предложенный алгоритм будет работать с непрямоугольными островами, а зигзагами или вот такой формы? [["1","1","1"],["0","1","0"],["1","1","1"]]

  • @Denis-rh9jp
    @Denis-rh9jp ปีที่แล้ว

    Сам не смог решить, после того как ты объяснил понял почти все, кроме того как можно передать в квадратных скобочках два индекса 0_о

  • @romansoleynik7899
    @romansoleynik7899 3 ปีที่แล้ว

    Сделай пжл отдельный канал по разбору задач или совмести их в отдельный плейлист

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว +1

      :) Такой плейлист у нас уже есть: th-cam.com/play/PL0k-9Y7O1GwccXKHRzmvVj17yB7T9pjTo.html

    • @romansoleynik7899
      @romansoleynik7899 3 ปีที่แล้ว

      @@frontendscience 🔥 Спасибо

  • @SwankyOfficial
    @SwankyOfficial 3 ปีที่แล้ว

    хоть я и питонист, но разве так же с колонкой не надо делать проверку, как со строкой?) Т.к. можно стоять на 0:0 и искать в C-1 (-1). Ну а решал бы так же, но при этом я бы отправлял координаты в отдельный список, а не заменял 6ками. Так бы у меня в будущем были вариации по расширению ;)

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว

      с колонкой не надо - так как в этом случае просто вернется undefined, и не будет эксепшена.

  • @fetfrum
    @fetfrum 3 ปีที่แล้ว

    Что за фильм со Скалой? :)

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว +3

      Путешествие 2: Таинственный остров

  • @СергейМасленников-б3в
    @СергейМасленников-б3в 2 ปีที่แล้ว

    grid2: тот момент, когда между "островами" нет "воды"

  • @keksinjo
    @keksinjo 3 ปีที่แล้ว +1

    единственное, что не оч понравилось, что изначальный массив будет мутироваться

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว

      На литкод по умолчанию есть требование решить задачу не только наиболее оптимальным способом по времени, но и по памяти. Поэтому делается мутация. Это не продакшен код. А решение задачи с заданными условиями в сендбоксе.

  • @talivel118
    @talivel118 3 ปีที่แล้ว

    ВОПРОС К SENIOR, вы вообще смоги решить эту задачу, если да то за сколько времени вас удалось это сделать?

    • @ihorkovryhin1767
      @ihorkovryhin1767 3 ปีที่แล้ว

      Синьор такую задачу должен писать с листа минут за 5-10 это максимум.

  • @evgeniiAn
    @evgeniiAn 3 ปีที่แล้ว

    Подобная задача в Яндексе, при отборе на стажировку. Только там еще надо острова распределить по категориям, в зависимости от формы.

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว +1

      Класс! Они там четко фиксированных форм бывают что ли?

    • @evgeniiAn
      @evgeniiAn 3 ปีที่แล้ว

      @@frontendscience Да нет, просто надо определить, какие острова занимают всю высоту матрицы, а какие только часть)

  • @neverblued
    @neverblued 3 ปีที่แล้ว

    Хорошо бы ещё обойтись без дублирования кода в обработке соседей.

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว +4

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

  • @dasvas9383
    @dasvas9383 3 ปีที่แล้ว +1

    Но вы же мутируете массив с данными...

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว

      Почитай требования к задаче...

  • @DevilAlex03
    @DevilAlex03 3 ปีที่แล้ว

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

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว +2

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

    • @DevilAlex03
      @DevilAlex03 3 ปีที่แล้ว

      @@frontendscience Тут я попался, верно. Поправил!

    • @DevilAlex03
      @DevilAlex03 3 ปีที่แล้ว

      @@frontendscience насчет условия, не совсем понял. Мутация матрицы ведь необязательна, если использовать, например, мапу или объект для хранения пройденных ячеек (доступ к элементу в таком случае будет О(1))?
      Если я ошибаюсь, то поправьте пожалуйста

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว

      @@DevilAlex03 Все верно, но на leetcode во всех задачах есть есть еще условие реализовать алгоритм как можно более оптимальней и по памяти. Именно поэтому без надобности никто не делаем копию массива и жертвует мутацией в угоду памяти. Это не продакшен эвайромент а именно вот такой сендбокс со своими условиями.

    • @DevilAlex03
      @DevilAlex03 3 ปีที่แล้ว

      @@frontendscience Теперь понимаю, впервые столкнулся с этим сайтом, спасибо за разъяснение

  • @user-paint-alexandra
    @user-paint-alexandra 3 ปีที่แล้ว

    Чуть по-другому сделала, хотя с рекурсией интереснее:
    let grid1 = [
    [1, 1, 1, 1, 0, 0],
    [1, 0, 0, 1, 0, 0],
    [1, 1, 0, 0, 0, 0],
    [0, 0, 0, 0, 1, 1],
    ]
    let obj = {}
    let checkCoord = (x, y, grid) => {
    if (grid[x][y] === 1) {
    if (Object.keys(obj).length) {
    let ifAHalfOfIcland = -1
    Object.keys(obj).forEach(key => {
    obj[key].forEach(el => {
    if (el.x - 1 === x && el.y === y) {
    ifAHalfOfIcland = key
    }
    if (el.x + 1 === x && el.y === y) {
    ifAHalfOfIcland = key
    }
    if (el.x === x && el.y + 1 === y) {
    ifAHalfOfIcland = key
    }
    if (el.x === x && el.y - 1 === y) {
    ifAHalfOfIcland = key
    }
    })
    })
    if (ifAHalfOfIcland > -1) {
    obj[ifAHalfOfIcland].push({x, y})
    } else {
    obj[Object.keys(obj).length] = [{x, y}]
    }
    } else {
    obj[0] = [{x, y}]
    }
    }
    }
    let numIslands = function(grid) {
    grid.forEach((item, x) => {
    item.forEach((row, y) => {
    checkCoord(x, y, grid)
    })
    });
    return Object.keys(obj).length
    }
    console.log(numIslands(grid1))

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว

      Благодарю за решение. К сожалению что-то пошло не так и выдает не верный ответ на вот таком кейсе:
      [["1","1","1","1","0"],["1","1","0","1","0"],["1","1","0","0","0"],["0","0","0","0","0"]]
      Output 0
      Expected 1

  • @eugenenibiru2020
    @eugenenibiru2020 3 ปีที่แล้ว +1

    Мозги скрипят(

  • @andreipopov2700
    @andreipopov2700 2 ปีที่แล้ว

    +++

  • @Denis-rh9jp
    @Denis-rh9jp ปีที่แล้ว

    grid[r][c]

  • @МаксимСоловьев-с9н
    @МаксимСоловьев-с9н ปีที่แล้ว

    const input1 = [
    ["1", "0", "1", "0", "1"],
    ["1", "1", "0", "1", "0"],
    ["1", "1", "0", "0", "0"],
    ["0", "0", "0", "0", "0"],
    ];
    const mathIsland = (arg) => {
    let count = 0;
    let onIsland = false;
    for (let i = 0; i < arg.length; i++) {
    for (let j = 0; j < arg[i].length; j++) {
    if (j === 0) {
    onIsland = false;
    }
    if (arg[i][j] === "0" && onIsland === true) {
    onIsland = false;
    }
    if (i === 0) {
    if (arg[i][j] === "1" && onIsland === false) {
    count++;
    onIsland = true;
    }
    if (arg[i][j - 1] === "0" && arg[i][j] === "1" && onIsland === false) {
    count++;
    onIsland = true;
    }
    } else {
    if (arg[i][j] === "1" && onIsland === false) {
    if (arg[i - 1][j] !== "1") {
    count++;
    }
    onIsland = true;
    }
    if (arg[i][j - 1] === "0" && arg[i][j] === "1" && onIsland === false) {
    if (arg[i - 1][j] !== "1") {
    count++;
    }
    onIsland = true;
    }
    }
    }
    }
    return count;
    };
    console.log(input1[0], input1[1], input1[2], input1[3]);
    console.log(mathIsland(input1));

  • @alexErtemenko
    @alexErtemenko 3 ปีที่แล้ว

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

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว +1

      Так присылай решение, посмотрим насколько оно проще

    • @hsmtrue
      @hsmtrue 3 ปีที่แล้ว

      @@frontendscience я прислал свое, похоже на то что тут описали, посмотрите. Отдельным комментарием.

    • @frontendscience
      @frontendscience  3 ปีที่แล้ว

      @@hsmtrue Посмотрел) Не подошло. Под решением написал почему.