How to find substring palindrome? Task from frontend job interview | LeetCode | JavaScript

  Рет қаралды 16,369

Front-end Science with Sergey Puzankov

Front-end Science with Sergey Puzankov

Күн бұрын

Пікірлер: 109
@frontendscience
@frontendscience 4 жыл бұрын
Таймкоды: 00:00 Начало. 00:26 Уровень сложности на leetcode. 00:45 Разбираем условие задачи. 01:42 Зевнул🙂 02:29 Разбираем алгоритм решения. 05:42 Пишем код решения. 06:30 Пишем вспомогательную функцию expandFromCenter. 08:58 Находим максимальную длину палиндрома для каждого символа в строке. 10:58 Оптимизируем сложность алгоритма по памяти. 13:04 Проверяем работу алгоритма - запускаем!. 13:42 Сложность алгоритма. 14:01 Алгоритм Манакера. 14:18 Присылайте свои решения.
@captain_kobi7476
@captain_kobi7476 3 жыл бұрын
Крутое видео
@frontendscience
@frontendscience 3 жыл бұрын
@@captain_kobi7476 Рад, что понравилось! :)
@vadavur
@vadavur 3 жыл бұрын
Здорово рассказано про память. Когда решал сам перед просмотром, алгоритм использовал схожий, но работал как раз-таки с подстроками. Плюс, пришлось вставить строчку-костылек на случай отсутствия палиндромов в строке. var longestPalindrome = function(s) { let longestPalindrome = ''; for (let i = 0; i < s.length - 1; i++){ if (s[i] === s[i + 1]) longestPalindrome = getLongestString(longestPalindrome, getPalindrome(s, i, i + 1)); if (s[i - 1] === s[i + 1]) longestPalindrome = getLongestString(longestPalindrome, getPalindrome(s, i - 1, i + 1)); } return getLongestString(longestPalindrome, s[0]); }; function getPalindrome(string, start, end) { while (string[start] && string[start] === string[end]) { start--; end++; } return string.slice(++start, end); } function getLongestString(string1, string2) { if (string1.length > string2.length) return string1; return string2; }
@frontendscience
@frontendscience 2 жыл бұрын
Классно вышло!
@oxanananieva2643
@oxanananieva2643 3 жыл бұрын
Cпасибо, очень полезный разбор задачи
@frontendscience
@frontendscience 3 жыл бұрын
И Вам спасибо, что смотрите
@evgeniychip
@evgeniychip 2 жыл бұрын
спасибо Сергей за контент, хорошо прокачался на литкоде благодаря тебе, так и до L5 фаанга недалеко, видосы смотрятся на одном дыхании
@oleksandrlevchenko4028
@oleksandrlevchenko4028 4 жыл бұрын
Спасибо вам, очень актуально для меня сейчас)
@frontendscience
@frontendscience 4 жыл бұрын
Рад, что пригодилось! ;)
@holus-bolus
@holus-bolus 2 жыл бұрын
Спасибо за такие видео. Смотрел одного английского блогера, он делал задачи с codewars, я чуть не уснул от его монотонности.
@maksbzr5655
@maksbzr5655 3 жыл бұрын
Очень крутая тема с решением задач
@frontendscience
@frontendscience 3 жыл бұрын
Благодарим за поддержку!
@stormd2902
@stormd2902 4 жыл бұрын
Спасибо за такие полезные видео Помогает
@frontendscience
@frontendscience 4 жыл бұрын
И Вам спасибо, что смотрите! :) Это вдохновляет)
@maksudibr8637
@maksudibr8637 3 жыл бұрын
Большой рахмат!
@frontendscience
@frontendscience 3 жыл бұрын
Ыраазы бул жакты
@dmytrokoka3796
@dmytrokoka3796 4 жыл бұрын
Хороший разбор! А если не получается, то не волнуйся, если не работает. Если бы все всегда работало, у тебя бы не было работы.
@romanryaboshtan9270
@romanryaboshtan9270 3 жыл бұрын
Я понял, как работает этот алгоритм, автор молодец
@ВадимБєк
@ВадимБєк 4 жыл бұрын
Было не просто но хоть как-то решить смог, изначально не было идей вобще но чуть глянув видео и услышав про 2 указателя подумал про два цикла и дальше делал сам, понимаю что алгоритм вышел нагруженным и в условие не попадаю но рад что смог хоть как-то сделать const longestPalindrome = str => { let palindromes = {}; for(let i = 0; i0;j--){ if(str[i] == str[j]){ let check = str.slice(i,1+j); let isPalindromebool = true; [...check].forEach((elem,index,array)=>{ elem == array[array.length - index - 1] ? elem : isPalindromebool = false; if(index == array.length -1 && isPalindromebool == true)palindromes[check.length] = str.slice(i,1+j); }) } } } let result = Object.values(palindromes); return result.sort((a,b)=> a-b)[0]; } Если есть советы по тому как можно улучшить код буду рад услышать их Теперь можно и решение посмотреть.
@frontendscience
@frontendscience 4 жыл бұрын
Благодарю за решение! Идея интересная вышла. Собираешь все возможные палиндромы. Из оптимизации я бы наверное предложил хранить максимальный найденный до этого палиндром и если на новой итерации нашелся еще длиннее то заменять его в переменной. Просто сейчас все палиндромы (и его подчасти) хранятся в объекте и это большой удар по памяти ) PS: код надо подправить. В последних 2~ строках ошибка. вот так будет работать: let result = Object.keys(palindromes); return palindromes[result.sort((a,b)=> b-a)[0]];
@Kozmo969
@Kozmo969 Жыл бұрын
Крутое видео, спасибо большое!
@АрсаланБазаров-л4п
@АрсаланБазаров-л4п 2 жыл бұрын
Спасибо большое за годный контент!))
@serverizedinov7942
@serverizedinov7942 3 жыл бұрын
красава!
@multiply87
@multiply87 3 жыл бұрын
Спасибо, очень интересно и полезно)
@frontendscience
@frontendscience 3 жыл бұрын
Ты прям пачками, я смотрю, задачи проходишь! Крутецки!))) Высылай решения!
@multiply87
@multiply87 3 жыл бұрын
@@frontendscience прости, конкретно вчера я их не решал, а смотрел в качестве развлекательного видео. Очень нравится видеть алгоритмы, которые раньше и вообразить не мог)
@АрсаланБазаров-л4п
@АрсаланБазаров-л4п 2 жыл бұрын
thank you for your excellent content!))))
@zosyanax
@zosyanax 2 жыл бұрын
мой костылек на пыхе, с примерами Сергея работает, на остальном не тестил public function testPalindromSubstring( string $title ): string { $title = preg_replace( '/\W/', '', strtolower( $title ) ); $start = 0; $end = 0; if ( strlen( $title ) expand( $title, $i, $i + 1 ); } if ( $title[ $i + 1 ] === $title[ $i - 1 ] ) { $expand = $this->expand( $title, $i - 1, $i + 1 ); } if ( $expand[ 0 ] - $expand[ 1 ] >= $end - $start ) { [ $end, $start ] = $expand; } } return substr( $title, $start, $end ); } public function expand( string $title, int $start, int $end ): array { while ( $start >= 0 && $end < strlen( $title ) && $title[ $start ] === $title[ $end ] ) { $start--; $end++; } $start++; return [ $end - 1, $start ]; }
@ИванКущ-р7с
@ИванКущ-р7с 3 жыл бұрын
Просто лучший!!!
@ОлегСелин-ш9ы
@ОлегСелин-ш9ы 4 жыл бұрын
Включить в понедельник утром, в начале рабочего дня... Это было страшной ошибкой)))
@frontendscience
@frontendscience 4 жыл бұрын
🤓 ))
@Sleep88Walker
@Sleep88Walker 3 жыл бұрын
Что-то хорошее
@frontendscience
@frontendscience 3 жыл бұрын
Спасибо за поддержку
@EvilYou
@EvilYou 3 жыл бұрын
Задачу решил, но возникли интересные ситуации при отправке на leetcode. При первой проверке результат - Runtime error, так как в дополнительной функции, которая проверяет на палиндром, я преобразовывал строку в массив и обратно. При второй отправке оптимизировал эту функцию и leetcode зачел мое решение, но с весьма печальным результатом: Runtime: 6624 ms, faster than 5%. Ну и, наконец, при третьей, добавил выход из цикла когда получить палиндром длиннее было уже невозможно. Вот решение: function longestPalindrome(str) { let longest = null; let maxLength = 0; let current = ""; for (let i = 0; i < str.length; i++) { if (maxLength >= str.length - i) break; // если длиннее строки быть не может for (let j = i; j < str.length; j++) { current += str[j]; if (isPalindrome(current)) { if (current.length > maxLength) { maxLength = current.length; longest = current; } } } current = ""; } return longest; } function isPalindrome(s) { let half = Math.floor(s.length / 2); for (let i = 0; i < half; i++) { if (s[i] !== s[s.length - 1 - i]) return false; } return true; } Результат на leetcode: Runtime: 2004 ms, faster than 15.32% Конечно, решение далеко не из быстрых получилось. Поэтому сейчас начинаю смотреть решение из видео.
@frontendscience
@frontendscience 3 жыл бұрын
Классно! Круто что сам решил! Благодарю за вариант!
@EvilYou
@EvilYou Жыл бұрын
Суть решения: прохожусь циклом по строке и проверяю символы слева и справа. Если равны, проверяю дальше и тд. Второй вложенный цикл for делает то же самое, только если два символа встретились подряд. function longestPalindrome(s) { let result = s[0]; for (let i = 0; i < s.length; i++) { let currentStr = s[i]; let maxJ = Math.min(i + 1, s.length - i); // сколько до конца строки справа или слева if (result.length > maxJ * 2) break; // оптимизация, чтобы меньше проверять for (let j = 1; j < maxJ; j++) { let left = i - j; let right = i + j; if (s[left] !== s[right]) break; currentStr = s[left] + currentStr + s[right]; if (currentStr.length > result.length) { result = currentStr; } } if (s[i + 1] !== s[i]) continue; currentStr = ''; maxJ = Math.min(i + 1, s.length - 1 - i); for (let j = 0; j < maxJ; j++) { let left = i - j; let right = i + 1 + j; if (s[left] !== s[right]) break; currentStr = s[left] + currentStr + s[right]; if (currentStr.length > result.length) { result = currentStr; } } } return result; } На leetcode: Runtime: 105ms (Beats 59.34%). Решение из видео чуть быстрее: Runtime 100 ms.
@lycan9590
@lycan9590 Жыл бұрын
Что-то хорошее )
@sergeylysenko2866
@sergeylysenko2866 3 жыл бұрын
что-то хорошее
@Юлия-ш9э4п
@Юлия-ш9э4п 3 жыл бұрын
Круто
@vadimman6140
@vadimman6140 4 жыл бұрын
Увидел условия стало как-то не по себе, нус будем пробовать
@frontendscience
@frontendscience 4 жыл бұрын
Успехов! Присылай потом решение!
@wisarty
@wisarty 2 жыл бұрын
Дякую
@eghishemanukyan3021
@eghishemanukyan3021 2 жыл бұрын
thanks
@СергейКозлов-н1л
@СергейКозлов-н1л 4 жыл бұрын
Сергей, а не могли бы указать, какие структуры данных необходимо 100% знать фронтенд-разработчику? И что делать, если прочитал условие задачи, но нет каких-то мыслей с чего начать решать, т.е. по сути нет идеи? Откуда брать это понимание, какой алгоритм нужно использовать?
@frontendscience
@frontendscience 4 жыл бұрын
Олтичная идея для новго видео! :) Для фронтендера точно важно знать: строки, числа, объекты, массивы, сеты и мапы. Так как с этими структурами прийдется работать. А вот для прохождения суровых собеседований в большие компании понадобится знания еще таких структур: linked list, binary tree, graph, trie, stack, queue, min/max stack. Что касается самого алгоритма - то тут нужна практика. Надо начинать с задач попроще и повышать уровень сложности постепенно. С практикой прийдет понимание когда какой алгоритм использовать. Если есть задача которую не получается совсем решить - можно поискать решение (даже если на другом языке программирования будет) и просто посмотреть сам алгоритм. Но потом обязательно надо повторить самому и написать работающее решение.
@Юлия-з5м2й
@Юлия-з5м2й 4 жыл бұрын
Попробуй объяснить человеческим языком самому себе как бы ты это сделал как человек, чисто логически. Или уточке на утином) Разбей на подзадачи, совсем простые. Затем по пунктам приступай реализовывать уже в коде. И гугли по мере незнания конкретные моменты. Скорее всего идеи как написать нет на один маленький и совершенно определённый пункт подзадачи, который можно погуглить и применить к своему алгоритму решения.
@Юлия-з5м2й
@Юлия-з5м2й 4 жыл бұрын
Сначала будут выходить не очень быстрые и не самые изящные решения. Но по мере тренировки будешь уже видеть, что ага, с таким ты уже сталкивался и это можно покрутить так и сяк. И так найдёшь решение. Главное не останавливаться и не бросать
@arslanaliisaev3324
@arslanaliisaev3324 3 жыл бұрын
Ты лучший )
@alicenNorwood
@alicenNorwood 3 жыл бұрын
1:54 - ваааааа, я ровно так же делаю, когда надо резко вернуть фокус
@frontendscience
@frontendscience 3 жыл бұрын
)))) бро)
@maksymzhovtaniuk3958
@maksymzhovtaniuk3958 2 жыл бұрын
Спасибо за видео, очень интересное решение, даже не подумал, что так можно🤔 Кому интересно сделал примерно так (за 10-15 минут) const input1 = 'babad' // bab aba const input2 = 'cbbd' // bb const input3 = 'mississippi' // ississi const input4 = 'ac' // a c const longestPalindrome = function (str) { let maxCheckLength = 0 let tmp = 0 const palindrome = (text) => { text = text.toLowerCase().trim() return text === text.split('').reverse().join('') } for (let i = 0; i < str.length; i++) { for (let j = str.length - 1; j >= 0; j--) { if (str[i] === str[j]) { let check = str.slice(i, j + 1) tmp = check.length let checkResult = palindrome(check) if (checkResult && tmp >= maxCheckLength) { maxCheckLength = tmp console.log(`Result of checking string "${check}" ==== ${checkResult} with length ${maxCheckLength}`); } } } } return 0 } console.log(longestPalindrome(input1)) console.log(longestPalindrome(input2)) console.log(longestPalindrome(input3)) console.log(longestPalindrome(input4))
@Dobermun
@Dobermun 3 жыл бұрын
Вау
@userScKonov
@userScKonov 3 жыл бұрын
мне помогло)
@frontendscience
@frontendscience 3 жыл бұрын
Рад, что было полезно!
@dan1ar
@dan1ar 3 жыл бұрын
Это решение с ЛитКода, а я вот слышал, что можно решить через хеши и бинпоиск, вы знаете о таком алгоритме?
@frontendscience
@frontendscience 3 жыл бұрын
Это мое решение) На Литкод я беру само условие задачи. Не слышал и пока не представляю, как именно эту задачу можно решить через хеши и бинарный поиск. Бинарный поиск это поиск одного конкретного элемента, а нам необходимо найти подстроку, которая является палиндромом.
@АртемФролов-э2ч
@АртемФролов-э2ч 3 жыл бұрын
​@@frontendscience Здравствуйте! Тут идея про хеши и бин поиск в следующем: есть функция, у которой аргументы такие же, как и у Вашей expandFromCenter, только мы не проходимся циклом while, а используем бин поиск для нахождения самого длинного палиндрома, только я не совсем понимаю, как использовать хеши. Прикреплю псевдокод этой функции(тут она ищет кол-во палиндромов): int binarySearch(s : string, center, shift : int): //shift = 0 при поиске палиндрома нечётной длины, иначе shift = 1 int l = -1, r = min(center, s.length - center + shift), m = 0 while r - l != 1 m = l + (r - l) / 2 //reversed_hash возвращает хэш развернутой строки s if hash(s[center - m..center]) == reversed_hash(s[center + shift..center + shift + m]) l = m else r = m return r
@kirsan3103
@kirsan3103 3 жыл бұрын
4 утра это ж середина рабочего дня разве нет? о_О XD
@frontendscience
@frontendscience 3 жыл бұрын
😂
@АнастасияДанилюк-я7л
@АнастасияДанилюк-я7л 4 жыл бұрын
Вот такое есть(Не мое): function longest_palindrome(s) { var max = ''; for (var i = 0; i < s.length; i++) { for (var j = 0; j < 2; j++) { var left = i; var right = i + j; while (s[left] && s[left] === s[right]) { left--; right++; } if ((right - left - 1) > max.length) { max = s.substring(left + 1, right); } } } return max; }
@frontendscience
@frontendscience 4 жыл бұрын
Ну алгоритм точно такой же - просто немного по другому реализован. Я специально вынес часть кода в отдельную функцию и сделал 2 переменные len1 и len2 чтобы было проще понять. По времени сложность алгоритма та же, а вот по памяти тут O(n) так как хранится вся подстрока.
@frontendscience
@frontendscience 4 жыл бұрын
А теперь присылай свое решение ;)
@АнастасияДанилюк-я7л
@АнастасияДанилюк-я7л 4 жыл бұрын
@@frontendscience пока нос еще (у меня) не дорос. Понять что написано могу , но написать еще нет ... Доработать , подкорректировать
@АнастасияДанилюк-я7л
@АнастасияДанилюк-я7л 4 жыл бұрын
На codwerse эта задача 4 ku уровня .Из 4 уровня удалось решить пока 3 задачи.
@das8434
@das8434 3 жыл бұрын
А как программа понимает что мы хотим от аргументов L, R ? Как она понимает, что под L и R мы подразумеваем указатель лево и право в строке ?
@frontendscience
@frontendscience 3 жыл бұрын
Мы ведь сами задаем индексы элементов для этих указателей. Для L мы задаем индекс элемента слева, а для R - индекс элемента справа.
@das8434
@das8434 3 жыл бұрын
@@frontendscience да но как мы задаём программе, что L это левый указатель, а R это правый ? В функции написано только что L >= 0, а R < s.length. Тоесть мы просто говорим программе что L меньше или равняется 0 и что R меньше длины s. Но мы не как не объясняем программе что L и R это правый и левый указатель в массиве и что они значат
@frontendscience
@frontendscience 3 жыл бұрын
Мы вызываем функцию expandFromCenter и передаем в нее уже готовые L и R указатели в качестве аргументов. (Строки 11 и 12). Дальше на каждой итерации Левый указатель бежит в лево пока не дойдет до начала, а правый бежит вправо - пока не дойдет до конца массива (это если элементы равны друг другу на каждой итерации).
@das8434
@das8434 3 жыл бұрын
@@frontendscience тоесть мы сначала пишем функцию expandFromCenter для указателей L и R, а позже ее подключаем к len1 и len 2 ?
@frontendscience
@frontendscience 3 жыл бұрын
@@das8434 да именно так!
@Юлия-з5м2й
@Юлия-з5м2й 4 жыл бұрын
спать очень важно для мозга)
@frontendscience
@frontendscience 4 жыл бұрын
- Вы хорошо высыпаетесь? - Куда?
@frontendscience
@frontendscience 4 жыл бұрын
- Доктор, это что вы мне такое классное дали? - Поспать!
@realkazakhit45
@realkazakhit45 Жыл бұрын
@UlyanaVorobeva
@UlyanaVorobeva 11 ай бұрын
А что делать, если надо найти не подстроку палиндром, а подпоследовательность, т е можно удалять символы в середине, лишь бы был палиндром?)
@verbychi3091
@verbychi3091 3 жыл бұрын
Недосмотрев до конца, считал, что знаю алгоритм Манакера... недолго я прожил во лжи
@frontendscience
@frontendscience 3 жыл бұрын
Смешно шутите :)
@dad2863
@dad2863 3 жыл бұрын
Почему при вводе hellow вывод llo
@redmex176
@redmex176 11 ай бұрын
вывод ll
@ilovemama6997
@ilovemama6997 3 жыл бұрын
проверь на "ааа" не работает
@frontendscience
@frontendscience 3 жыл бұрын
Смешно! А сам че не проверил? Ты удивишься - ведь все таки работает!
@alexey731
@alexey731 4 жыл бұрын
блин сложно..
@frontendscience
@frontendscience 4 жыл бұрын
Рекомендую, если еще не видели предыдущие 2 задачи про палиндромы начать именно с них. Тогда эта станет понятней. На будущее я планирую снять еще видео с алгоритмом 2 указателя - думаю это поможет лучше разобраться и с это задачей тоже.
@alexey731
@alexey731 4 жыл бұрын
@@frontendscience Спасибо большое, сейчас посмотрю! А так задача очень полезная, хотелось бы ее осмыслить
@frontendscience
@frontendscience 4 жыл бұрын
@@alexey731 Ну надеюсь я в видео все подробно разобрал, и что прийдет осмысление после просмотра предыдущих. :)
@geimproberegov7350
@geimproberegov7350 2 жыл бұрын
Помогите тупому учиться. ) я второй день не догоняю , как у нас передается в функции expandFromCenter наша середина, если у нас итерируется одни и те же элементы , так как мы передаем их через i . то есть L=0, R=0 и так далее. , почему к примеру в слове "babad" начинает while Отрабатывать именно на середине , если у нас и до середины элементы получается одинаковые a=a ..
@react_js
@react_js Жыл бұрын
как я понял, expandFromCenter не начинает с середины, а с самого начала. потому что цикл for начинает i с нуля. а дальше i прибавляется. а R++ и L- -. и таким образом дальше идет. Непонятно почему он в слове babad середину показывает. там же i с нуля идет. ну да ладно
@whiteguards43
@whiteguards43 Жыл бұрын
@@react_js так там услвоие L >= 0, если L === 0, то L-- это -1, что соотвественно прервет цикл while
@mikhailblush5261
@mikhailblush5261 3 жыл бұрын
отец, ты спишь вообще?
@frontendscience
@frontendscience 3 жыл бұрын
Я как медведь - накапливаю!
@eghishemanukyan3021
@eghishemanukyan3021 2 жыл бұрын
если ты смотришь это сообщение то пожалуйста оставь свое мнение об этом решение мне очень интересно const longestPalindrome = function(s){ let str = ''; let arr = []; let res = [] for(let i = 0; i < s.length; i++) { str = s[i]; for(let j = i+1; j < s.length; j++) { str += s[j] if(isPalindrome(str)) { arr.push(str) } } } arr.sort((a,b) => b.length - a.length); if(!arr.length) return s[0] // for(let i = 0; i < arr.length; i++) { // if( arr[0].length === arr[i].length ) { // res.push(arr[i]) // } // } return arr[0] }; function isPalindrome(s) { return s === s.split('').reverse().join('') }
@ТимофейБобыльков
@ТимофейБобыльков 3 жыл бұрын
Что-то хорошее
@frontendscience
@frontendscience 3 жыл бұрын
Благодарю! )
@pavlo2846
@pavlo2846 3 жыл бұрын
что-то хорошее
Numeric palindrome | Solve the LeetCode problem in JavaScript
11:17
Front-end Science із Сергієм Пузанковим
Рет қаралды 12 М.
How to randomly sort an array? | LeetCode task | JavaScript
10:20
Front-end Science із Сергієм Пузанковим
Рет қаралды 12 М.
Andro, ELMAN, TONI, MONA - Зари (Official Music Video)
2:50
RAAVA MUSIC
Рет қаралды 2 МЛН
요즘유행 찍는법
0:34
오마이비키 OMV
Рет қаралды 12 МЛН
LeetCode task about collecting rainwater | JavaScript interview
24:45
Front-end Science із Сергієм Пузанковим
Рет қаралды 20 М.
Solving the problem from JS interview - The valid sequence of brackets | LeetCode problems
15:46
Front-end Science із Сергієм Пузанковим
Рет қаралды 39 М.
Interview Task: Brick Wall | JavaScript
10:19
Front-end Science із Сергієм Пузанковим
Рет қаралды 22 М.
How to find two numbers in array that together will give a desired sum? | Sum of Two | JS
10:29
Front-end Science із Сергієм Пузанковим
Рет қаралды 22 М.
Manacher's Algorithm | Longest Palindromic Substring
21:47
Fluent Algorithms
Рет қаралды 31 М.
How to remove duplicates from a sorted array? | Leetcode task
9:42
Front-end Science із Сергієм Пузанковим
Рет қаралды 11 М.