SOLID: Принцип подстановки Барбары Лисков/ LSP (The Liskov Substitution Principle)

  Рет қаралды 119,239

Sergey Nemchinskiy

Sergey Nemchinskiy

Күн бұрын

Пікірлер: 441
@SergeyNemchinskiy
@SergeyNemchinskiy 3 ай бұрын
💪Новый поток advanced тренинга Enterprise patterns стартует 2.12 - go.foxminded.ua/4eXh83k
@-ebaka
@-ebaka 4 жыл бұрын
Роберт что-то непонятное сказал, то ли дело Барбара
@6598335
@6598335 4 жыл бұрын
XDXDXD
@СергО-л6ф
@СергО-л6ф 3 жыл бұрын
Математически чёткое описание всегда более точное, чем словесное описание. Например, Правила Ассоциативноти и Дистрибутивности и звучат лучше...
@vadimsokhatsky2748
@vadimsokhatsky2748 2 жыл бұрын
Сергей, спасибо! С вашими видео обучение программированию становится сплошным кайфом!
@alokym86
@alokym86 Жыл бұрын
Я знаю чому для Вас формулювання Барбари Лісков складне і заплутане! Все тому, що для Вас і квадрат - це не прямокутник!) Дякую за відео. Приклад з квадратом підірвав мій мозок.
@АлексейПоляков-ш1э
@АлексейПоляков-ш1э 4 жыл бұрын
TDD - ДА! Про return null - ДА! Спасибо за видео!
@Snake19S
@Snake19S 4 жыл бұрын
Сергей, вы такой живчик в этом видео. Отлично получилось!
@masterbedroom594
@masterbedroom594 4 жыл бұрын
Сергей, выражаю вам искреннюю благодарность
@whoiam6395
@whoiam6395 3 жыл бұрын
Спасибо огромное!!! В 3х словах рассказал то что я не смог понять ни в одном видео до этого.
@whoiam6395
@whoiam6395 3 жыл бұрын
А я их пересмотрел штук 10
@Virus-td9nc
@Virus-td9nc Ай бұрын
а я снова нихуя не понял
@YuretsUA
@YuretsUA 4 жыл бұрын
Очень доступное объяснение, особенно в разрезе примеров. Сразу вспомнил где я уже наговнокодил...
@БендерЗадунайский-щ9ы
@БендерЗадунайский-щ9ы 4 жыл бұрын
надо! про тест фёст надо про нуль из методов и ДОСКУ МАРКЕРНУЮ! НАДО!
@Asmaers
@Asmaers 4 жыл бұрын
и про красную футболку!
@VitaliyZlobin
@VitaliyZlobin 4 жыл бұрын
доску особенно, уши режет
@kspshnik
@kspshnik 3 жыл бұрын
дададад! и про TDD и про null надонадонадо!
@КазимБерловский
@КазимБерловский 3 жыл бұрын
Доска как раз под бумагой. Приглядись)
@TheHosuwisp
@TheHosuwisp 4 жыл бұрын
Математический вариант Барбары все разложил по полочкам, а то так много воды и ничего не понятно.
@SergeyNemchinskiy
@SergeyNemchinskiy 4 жыл бұрын
ахаха. Ну, рад за вас :)
@ffconsole3467
@ffconsole3467 4 жыл бұрын
Попався, математiк
@fakerliar6599
@fakerliar6599 4 жыл бұрын
Тоже объяснение Барбары показалось более простым и понятным, хотя не математик. Может это потому что я женщина?:D
@murationametisation4347
@murationametisation4347 3 жыл бұрын
Чувак, и мне показалось вариант Барбары намного ясный и простой. Но задело то, что автор видео так обобщает, "всем непонятно". Бро, как раз таки и понятно. Это может для гуманитариев проблема понять из-за боязни математики.
@lindx2533
@lindx2533 3 жыл бұрын
странно что ты за этим вариантом полез на ютьюб к немчинскому
@stakhovskiy
@stakhovskiy 4 жыл бұрын
Август месяц на дворе, а его все еще зовут Сергей Немчинский. Хосспаде! Спасибо за видео 😄Лайк.
@z1zzz
@z1zzz 4 жыл бұрын
А в чём прикол?
@igorost795
@igorost795 3 жыл бұрын
Попрошу! Сергей "тутромбик" Немчинский!
@НиколайЮрченко-о2й
@НиколайЮрченко-о2й 4 жыл бұрын
Классное видео, спасибо за детальное и простое объяснение, хотелось бы еще послушать про паттерн pipeline.
@BukasovMaxim
@BukasovMaxim 4 жыл бұрын
Я бы еще добавил, что при оверрайде неабстрактных методов, новый метод должен уметь принимать параметры в таком же диапазоне или шире (но ни в коем случае не уже), а при возвращении значений - в таком же диапазоне или уже (но ни в коем случае не шире). Например в каком-то классе есть метод, который умеет принимать число от 1 до 100 и возвращать от 1 до 10. Если его заоверрайдить так, что новый будет уметь принимать 0..200, а возвращать 3..5 - существующий клиентский код будет доволен. А вот если переписать так, что новый метод умеет принимать только 1..50 или вдруг возвращает 0..20 - то существующий клиентский код может быть неприятно удивлён. Также можно было бы еще сказать, что список выбрасываемых исключений можно сокращать, а исключения - конкретизировать, а вот наоборот - добавлять новые или расширять существующие - нельзя. Но с контролируемыми исключениями компилятор Джавы в этом плане сам подскажет, что можно, а что нет, а неконтролируемые... на то они и неконтролируемые :)
@vladzaiko5012
@vladzaiko5012 3 жыл бұрын
что то я не уверен что вы правильно написалали, если базовый класс возвращал от 1 до 10, а наследник будет возвращать от 3 до 5 а это ведь нарушение принципов о которых говорит Сергей - наследник не может возвращать меньше чем базовый класс. Базовый класс будет ожидать что при определенных параметрах вернется 10, а наследник вернет максимум 5 и все сломается.
@BukasovMaxim
@BukasovMaxim 3 жыл бұрын
@@vladzaiko5012 Представьте, что есть клиентский код, который умеет разговаривать на английском, французском и немецком. Есть базовый компонент, который иногда выдаёт клиенту что-то на английском, а иногда - на немецком. Клиент понимает эти сообщения от компонента. Если заменим этот компонент на новый, который будет выдавать сообщения только на английском (т.е. сузим диапазон возвращаемых значений), то клиентский код всё еще будет понимать эти сообщения. А вот если новый компонент вдруг заговорит на японском (т.е. расширим диапазон возвращаемых значений), то клиентский код сломается - потому что клиентский код не знает, что делать с сообщениями на японском.
@vladzaiko5012
@vladzaiko5012 3 жыл бұрын
@@BukasovMaxim так в том то и дело что клиентский код, который работает с базовым классом он не знает о функционале на японском языке, который уже появился в наследниках, он не может его вызвать и сломаться.
@BukasovMaxim
@BukasovMaxim 3 жыл бұрын
@@vladzaiko5012 Речь не о новых методах. Речь о старых методах, которые в новых компонентах вдруг начали выдавать на выход то, что старые компоненты не выдавали. Пример: батарейка (как компонент). Допустим у нас есть электрическая схема (клиент), которой нужно питание +5В +/-10% (т.е. от 4.5 до 5.5 - будет работать). Если у нас есть батарейка, которая выдаёт +5В +/- 5% (т.е. от 4.75 до 5.25), то все ОК, схема будет работать и не сгорит (потому что при замене компонентов выходные диапазоны можно сужать без проблем). Но если мы вставим батарейку +5В +/-20%, то она может дать или слишком мало (и схема не заработает) или слишком много (и схема сгорит). Поэтому, при подмене компонентов, расширять возвращаемый диапазон - нельзя (можно только сужать). Входной диапазон - наоборот: сужать нельзя (чтобы клиент не послал в компонент что-то такое, что он не поймёт), а расширять - можно (чтобы компонент был не глупее, чем его предок).
@catmother8368
@catmother8368 3 жыл бұрын
@@BukasovMaxim спасибо за отличное объяснение! А что, если у нас есть класс User и класс Admin, который является наследником от User? Допустим, в User есть метод, вохвращающий все права пользователей. User будет возвращать стандартные разрешения типа редактировать аккаунт и проч. Но у Admin нужно будет расширять этот список. Верно ли я понимаю, что в данном случае произойдёт нарушение принципа LSP? Как этого избежать? Не лучше ли выделить отдельный класс с правами, чтобы не нарушать SRP?
@protchenko.oleksandr
@protchenko.oleksandr 2 жыл бұрын
Або це просто вже 100те відео про Лісков і я нарешті зрозумів, або це найкраще з тих відео що я бачив і я зрозумів =))
@SergeyNemchinskiy
@SergeyNemchinskiy 2 жыл бұрын
я просто умею объяснять :)
@РостикНазаренко-п8с
@РостикНазаренко-п8с 2 жыл бұрын
Дуже вам дякую за пояснення. Все просто і зрозуміло )
@LeoMrakobes
@LeoMrakobes 4 жыл бұрын
@Sergey Nemchinskiy уже раньше обещал про "почему нельзя возвращать null"
@samuro2ua
@samuro2ua 4 жыл бұрын
Сергей, тема разработки через тестирование очень интересна! Как и SOLID. Спасибо.
@newprim1
@newprim1 3 жыл бұрын
Это самое лучшее объяснение принципа Лисков, что я встречал! Подписка однозначно
@bumer7721
@bumer7721 4 жыл бұрын
Об ACID интересно послушать. А за SOLID благодарю)
@maxlich9139
@maxlich9139 4 жыл бұрын
было, Владимир Кузнецов рассказывал.
@bumer7721
@bumer7721 4 жыл бұрын
@@maxlich9139 Дякую! Шукатиму.
@ntvisigoth
@ntvisigoth 4 жыл бұрын
@Sergey Nemchinskiy: Мужик, здоровья тебе! Отличное пояснение! ;)
@Madeniath
@Madeniath 4 жыл бұрын
Про TDD интересно было бы посмотреть.
@Oleksii_Leshchenko
@Oleksii_Leshchenko 4 жыл бұрын
про то, как оно в жизни
@NikolayMishin
@NikolayMishin 3 жыл бұрын
Самый сложный принцип, я вечно сыплюсь на нем))) но объяснение с прямоугольником и квадратом просто огонь! Спасибо 🙏
@yuriytheone
@yuriytheone Жыл бұрын
Ты сыпишься, потому что Solid ложны. Каждый принцип ложен! Давай про Srp. Когда ты создаёшь дочерний класс, ты его создаёшь именно таким, какой нужен именно в твоём коде. И именно с таким интерфейсом, который нужен твоему софту. Далее Ocp - ты не только можешь менять код метода, но и должен если это нужно для оптимизации. Да, представь себе Ocp тоже ложен. Далее Lsp - зачем тебе порождать наследников и закидывать в какой либо метод работающий с классом родителем. Ну, вот ты и не можешь объяснить... У немчи6ского яйц нет признать, Solid - фуфло.
@romansharpe1131
@romansharpe1131 4 жыл бұрын
Есть 2 класса Pistol и Rifle. Оба должны стрелять и перезаряжаться, значит выносим эту логику в класс Weapon и наследуемся от него. Появлятся новое оружие Knife, которое тоже атакует (стреляет) и также наследуется от Weapon. Теперь нож и стреляет, и перезаряжается. Реализуем LSP: между классами Weapon -> Pistol/Rifle делаем прослойку Gun и выносим туда логику перезарядки, а в Weapon оставляем только атаку/стрельбу. В итоге получаем: Weapon -> Gun -> Pistol/Rifle и Weapon -> Knife
@JackFastGame
@JackFastGame 4 жыл бұрын
Что делать, если к такому пришли только после написания кода? Например, нужно было сделать пистолет и ружьё, написали классы: Weapon -> Pistol, Weapon -> Rifle, а потом заказчик добавил в ТЗ нож. Что теперь делать?
@romansharpe1131
@romansharpe1131 4 жыл бұрын
​@@JackFastGame по нормальному неплохо было бы продумать все это заранее. Здесь же я привел свой пример, как выглядит LSP на практике.
@0imax
@0imax 4 жыл бұрын
@@JackFastGame Если код был написан правильно, то другие классы обращались к оружию через интерфейс Weapon и знать не знали, что там внутри. Тогда, если расширить структуру наследования от Weapon, как в комментарии выше, для внешнего кода работа этого класса не изменится.
@sergekim
@sergekim 4 жыл бұрын
Спасибо большое, Сергей!)
@alexeydeyev4970
@alexeydeyev4970 4 жыл бұрын
Очень интересно про Test First!!! В топ!!
@UnrealSPh
@UnrealSPh 4 жыл бұрын
Возможно, описание самой Барбары станет более понятным, если понимать когда она его предложила и как выглядели ассоциации типов в ЯП в то время, так как *спойлер* наследование не единственный способ ассоциации классов, а скорее частный случай. Насчёт использования наследования вы скорее всего заблуждаетесь, но причина заблуждения это легаси информация, которую из года в год перевыпускают в требованиях к коду. Если вы боретесь с копипастингом кода в проекте, то для этого больше подходит композиция кода. А если и методология разработки у вас не waterfall, то n-ступенчатая иерархия наследования превратит процесс изменения кода в ад (не забываем, что классы содержат ещё и стейты). Наследование сильный инструмент, но его главное назначение в продуктах, которые должны иметь жёсткую структуру иерархии, а так же максимально долгое время жизни самой иерархии. Напрмер фреймворки. Там наследование чувствует себя хорошо.
@AnnaIsHere
@AnnaIsHere 4 жыл бұрын
Хотелось бы больший упор на "как надо" вместо "как не надо"
@Gusto20000
@Gusto20000 3 жыл бұрын
Просто берете как не надо и делаете не так и будет как надо
@fractalzombie
@fractalzombie 3 жыл бұрын
С опытом придёт.
@АнтонДудкевич
@АнтонДудкевич 3 жыл бұрын
@@Gusto20000 и если это приходится объяснять, то, наверное, лучше никак не надо ;)
@screamer8932
@screamer8932 2 жыл бұрын
@@Gusto20000 Часто бывает 101 способ как не надо и только парочку как надо. Кск би так убегая от плохого 98 способа не попасть в 86))
@НиколайКоваленко-г2к
@НиколайКоваленко-г2к 8 ай бұрын
что касаемо lsp то что бы его не нарушать нужно просто придерживаться ооп. Надо постараться что бы его нарушить.
@topalidinka
@topalidinka 3 жыл бұрын
Я вас обожаю. Вы один из самых интересных и умных людей, которых я знаю) спасибо за знания и ваш юмор ☺️
@kotanhp3115
@kotanhp3115 2 жыл бұрын
Хорошее видео, жду такое же про Оксимирона
@vladyarets3596
@vladyarets3596 3 жыл бұрын
1:16 ну нафиг)) Как это прочитать получилось?!) 1:50 а не. вот тут нафиг, дяде Бобу спасибо огромнейшее!
@nataliia9346
@nataliia9346 4 жыл бұрын
Хотелось бы с таким классным объяснением и примерами послушать о Паттернах Программирования :)
@SergeyNemchinskiy
@SergeyNemchinskiy 4 жыл бұрын
эм.... Ну вот же оно: foxminded.com.ua/grasp-gof-design-patterns-advanced-on-line-course/
@xdef42
@xdef42 4 жыл бұрын
Я не использую наследование, я имплементирую интерфейсы, и занимаюсь агрегацией и композицией
@nanvlad
@nanvlad 4 жыл бұрын
Ты сраный рукожоп по версии Немчинского)) Я тоже агрегирую и композирую с интерфейсами + ещё дженерики
@alexeydeyev4970
@alexeydeyev4970 4 жыл бұрын
Имплементирование можно с натяжкой отнести к наследованию.
@grommaks
@grommaks 4 жыл бұрын
аналогично
@makaroningable
@makaroningable 4 жыл бұрын
Я наследую только абстрактные классы где имплементирую основное поведение, частично переопределяемое в классах наследниках. Но вот так чтоб в чистом виде унаследоваться от другого конкретного класса - такого тоже нет.
@woodzimierz9621
@woodzimierz9621 4 жыл бұрын
@@makaroningable есть товарищи, которые отрицают любую форму наследования
@ВадимКорженко-э8б
@ВадимКорженко-э8б 4 жыл бұрын
TDD интересно. С удовольствием послушаю.
@andrey7829
@andrey7829 11 ай бұрын
Очень понятно объясняете - супер, спасибо
@eligolin9947
@eligolin9947 2 жыл бұрын
По моей оценке вся суть проблемы была не раскрыта. Наследование как "анти паттерн", основана на сложности соблюдение LSP принципа. Пройдёмся по примерам в этом ролике: 1. Здесь Сергей рассказывает как плохо нарушать LSP принцип и к чему это может привести. Но дело не в том что его нельзя явно нарушать (это и так понятно), а в том что его очень сложно соблюдать. 2. Сергей здесь спасибо вам за альтернативное определение LSP (оно добавляет глубины понимания!) хотя по моей оценке примеры всё равно слабы. 3. По моей оценке этот пример вообще не об этом. Здесь проблема не в соблюдении или несоблюдении LSP, а в отсутствии инкапсуляции. Это был хороший пример того что лучше писать иммутабильные классы а setters/getters могут раскрыть зачастую внутреннюю структуру класса. С точки зрения наследования - это как раз один из немногих примеров где наследование является корректным. Наследования квадрата от прямоугольника является математически правильным и хорошо обоснованным с точки зрения качеств квадрата по отношению к качествам прямоугольника.Если бы состояние этих классов задавалось только при помощи конструктора, то никакой проблемы не было бы.
@yuriyfedoryshyn5206
@yuriyfedoryshyn5206 4 жыл бұрын
looking forward to a TDD video))
@sick_bear
@sick_bear 2 жыл бұрын
Наконец-то я понял этот принцип)
@wandos777
@wandos777 2 жыл бұрын
Супер, Сергей) Спасибо большое за видео, однозначно лайк!)
@torrvic1156
@torrvic1156 3 жыл бұрын
Я просто офигеваю от того, какой Сергей потрясающий интеллектуал. Мне страшно смотреть его видео. Страшно от того, что голова может лопнуть от обилия деталей, но потом трясущейся рукой тянусь и нажимаю кнопку начала просмотра, а потом инфа кое-как укладывается под отличную и ровную мелодию под методичный голос Сергея… P.S. короче, как тупой я понял принцип LSP главным образом в том, что не должен наследуемый класс делать больше, чем его основной класс.
@sfoxer
@sfoxer 4 жыл бұрын
Лучший препод =) Что бы мы без вас делали)
@РинатРаот
@РинатРаот 4 жыл бұрын
Ах**нно) наконец то понял в чём фишка этого принципа
@demidovmaxim1008
@demidovmaxim1008 4 жыл бұрын
Большое спасибо за выпуск!!!
@dmitriykozyrev8835
@dmitriykozyrev8835 4 жыл бұрын
Сразу лайк, спасибо тебе большое!
@ПашаХЗ-м8й
@ПашаХЗ-м8й 4 жыл бұрын
Давно ждал!
@alexanderlex-s933
@alexanderlex-s933 4 жыл бұрын
Изучаю Java с нуля. Даром прошли 5 лет в 1С, ООП - это взрыв мозга )))) 11:20 - я не могу использовать ООП в 1С, там только процедурный код - и да, я чувствую себя рукожопом, но за хорошую зарплату.
@vanilla1006
@vanilla1006 4 жыл бұрын
TDD, буду рад выслушать
@chat_mayevskogo
@chat_mayevskogo 3 жыл бұрын
Я себе тоже такую футболочку взял, тока черную, чтоб прям в тему последних времен было) Это когда джаву в js тоже будут компилить, наверное)
@woodzimierz9621
@woodzimierz9621 4 жыл бұрын
Про TDD очень интересно.
@jossefal1957
@jossefal1957 4 жыл бұрын
Хорошее объяснение, как обычно лайк
@kyryloshamraiev6289
@kyryloshamraiev6289 2 жыл бұрын
Очень наглядное объяснение. Проще говоря, нужно наследовать большее от меньшего, а не наоборот. Все станет на свои места, если считать, что квадрат - это частный случай прямоугольника, а не его наследник. Именно такой подход применят учёные. Потому обьяснение Барбары и не понятно программистам с ООП профдеформацией. :)
@RomanL321
@RomanL321 Жыл бұрын
ООП профдеформация это интересно))) всё под неё пытаются подтянуть, иногда усложняя код до нельзя...
@samirsalimkhanov3554
@samirsalimkhanov3554 4 жыл бұрын
Надо убрать бумагу между маркером и доской, потом писать. Я так думаю))) спаисбо за видос
@YuretsUA
@YuretsUA 4 жыл бұрын
Это нарушает 1-й принцип SLP, каждому объекту своя зона ответственности, доска для того, чтобы бумагу удерживать, и бумага не рвалась под фломастером. А бумага уже для отображения текстовой информации, так что тут все грамотно...
@СанжарСовет-ъ5ц
@СанжарСовет-ъ5ц 3 жыл бұрын
ну круто я понял как делать не надо ,а как делать надо сам разберусь спасибо лайк щищ
@dmitriimolchanov4971
@dmitriimolchanov4971 3 жыл бұрын
На примере mock, можно ведь для неиспользуемых методов оставить имплементацию парент класса, если это, конечно, не абстрактый класс или мы не имплеменим интерфейс. Хотя тут уже тогда нарушение как раз второго принципа-необходимо использовать интерфейс, или я что-то упустил?)
@СергейКарпиченко-с6б
@СергейКарпиченко-с6б 4 жыл бұрын
Спасибо большое за видео!
@Дмитрий-х2й5р
@Дмитрий-х2й5р 4 жыл бұрын
Спасибо. Только с квадратом как быть? Разве при наследовании у него не разумно переопределить метод подсчета площади?
@danilakapitanov7044
@danilakapitanov7044 3 жыл бұрын
Проблема не столько с определением площади, сколько с установкой сторон в клиентском коде. Правильным решением является создание абстрактного класса Figure с методом вычисления площади, от которого наследуется отдельно Rect и отдельно Square и реализуют метод вычисления площади. При этом для клиентского кода логика работы по установке сторон Rect и Square становится разная. Т.е. прямоугольник и квадрат по сути это разные сущности, как ,например, прямоугольник и круг.
@alexanderbelov6892
@alexanderbelov6892 3 жыл бұрын
@@danilakapitanov7044 Раз уж речь про клиентский код работы с фигурами, то характеристики фигур должны прописываться при создании, и отсутствовать методы их изменения, которые будут разными для разных фигур. К примеру для фигур могут быть методы get_square() для площади, и scale() для изменения размера. Но никак не изменения характеристик описывающих фигуру по отдельности. Для изменения фигуры с другими сторонами нужно создать новую фигуру, а предыдущую удалить.
@chocolazerboom7389
@chocolazerboom7389 4 жыл бұрын
Чёт прямо больно, что такие листы бумаги тратятся на то, что можно нарисовать на доске и стереть
@ESTechnonet
@ESTechnonet Жыл бұрын
А еще больно - звук маркера скрипучий.
@Zlobusz
@Zlobusz 4 жыл бұрын
Программирую на php в Symfony. На самом деле крайне редко приходится использовать наследование. Чаще фреймворк предоставляют интерфейсы, чем абстрактные классы. А наследоваться от какой-то конкретной реализации - тоже не совсем приятно. Поэтому случаев наследования можно по пальцам одной руки пересчить. Но принцип постановки Лисковой зашёл. Спасибо! Теперь бы про инверсию зависимости послушать)
@alexeyb5830
@alexeyb5830 2 жыл бұрын
потому, что ооп это не про наследование. В чистой архитектуре про это написано. Ооп это про полиморфизм. Старину Боба видимо не всего Немчинский читал, либо читал и не понял.
@Ardolynk
@Ardolynk Жыл бұрын
Разве mock не должен лежать в модуле test, недосягаемом для основного кода?
@user-hv8rh8nk9d
@user-hv8rh8nk9d 4 жыл бұрын
Отлично. Про тесты видео нужно
@kosatchev
@kosatchev 4 жыл бұрын
Принцип на примере семьи Хилтон: class Hilton { Money manageHotel(Hotel h) { Money m = makeCustomersHappy(c); return m; } } class Paris extends Hilton { @Override Money manageHotel(Hotel h){ Money m = sellHotel(h); return m; } }
@torrvic1156
@torrvic1156 3 жыл бұрын
Отличная шутка про ведение бизнеса 😂
@victortamanov
@victortamanov 4 жыл бұрын
Про TDD интересно будет посмотреть.
@alexeiceglei9841
@alexeiceglei9841 4 жыл бұрын
Юнит тесты интересны очень, особенно какие то хитрости, типо того же test first
@halgerdka9287
@halgerdka9287 4 жыл бұрын
про пример с квадратом. что мешает переопределить только метод getSquare() return a*a ? и вообще не трогать b
@АнтонДудкевич
@АнтонДудкевич 3 жыл бұрын
Я вот чет не понял, что плохого в примере с квадратом ) Ну, перезаписал он сторону с 10 на 5, ну, посчитал он площадь 25. Так она такая и есть в его случае ) Одинаковый интерфейс родителя и наследника никак не означает одинакового результата. Если он одинаковый всегда, - нахрена наследовали-то? )
@ivanoviv1844
@ivanoviv1844 11 ай бұрын
Ну как тут не подписаться ну никак и лайк поствлю и все что угодно
@TheDaslord
@TheDaslord 4 жыл бұрын
Сергей, спасибо за видео! Пожалуйста, снимите видео про возврат null, очень интересно, сам постоянно так делаю.
@dmitrinikolaev6986
@dmitrinikolaev6986 4 жыл бұрын
Про TDD было бы очень интересно послушать!
@igor_knv
@igor_knv 4 жыл бұрын
Понравился пример про квадрат и прямоугольник. Кто-кого расширяет.
@AAAnatoly
@AAAnatoly 4 жыл бұрын
Юнит-тесты очень интересны. По сути, я самоучка, и до юнит-тестов не дошёл, а начинать страшно
@yatigrrrrrrr
@yatigrrrrrrr 4 жыл бұрын
Такая же ситуация, но успокаиваю себя тем что джуну вроде как не обязательно наличие в стеке этой технологии и в будущем я смогу уже на реальном проекте это изучить
@k_v_i_i
@k_v_i_i 4 жыл бұрын
Ничего там нет страшного
@pgriggs1299
@pgriggs1299 4 жыл бұрын
@@yatigrrrrrrr Mockito и JUnit обязательно нужно знать джуну, также будет в разы круче, если код, например, в пет-проектах будет протестирован, потому что не протестированный код = нерабочий код
@r2d2925
@r2d2925 4 жыл бұрын
Что сложного, вызвал метод, передал тестов данные, и сравнял с заранее верным ответом. Это примитив, но смысл таков. Модульные тесты сложнее, но тоже эмалируешь роботу кода и сравниваешь с подготовленным ответом.
@John_602nd
@John_602nd 4 жыл бұрын
Ничего страшного совершенно, с ними другая проблема, что их ленятся писать. В Java все юнит тесты: @Before, @Test, assert(Equals/That/...) Если прочитать как это работает, то... Это в целом всё. А прочитать можно в первой же статье в гугле по запросу "Java UnitTest"
@VoroninPavel
@VoroninPavel 4 жыл бұрын
Хе-хе, люблю этот пример с квадратом и прямоугольником. Только он еще веселее. Если ввести иммутабельность с втиле старого доброго ФП и после вызова любого "мутирующего" метода возвращать экземпляро нового объекта, то квадрат-таки можно сделать проямоугольником, не нарушая принципы ООП ;-)
@zapplay
@zapplay 4 жыл бұрын
Хотелось бы увидеть видео о том почему нельзя возвращать null
@maxlich9139
@maxlich9139 4 жыл бұрын
Клевое видео. Единственное что Сергей не рассказал: что тогда делать-то, если твоя реализация не поддерживает данный метод базового класса? Я такое видел и не только в тестах и моках (и там да, бросались эксепшены; и вроде бы это даже было в каких-то известных либах)
@eb6006
@eb6006 4 жыл бұрын
Про TDD интересно, запишите пож-ста
@AndriySydorka
@AndriySydorka 4 жыл бұрын
Дядя Сережа, росскажи про Test first!
@homo-ergaster
@homo-ergaster 4 жыл бұрын
Вот что интересно, базовые классы Java нарушают LSP совершенно без стеснения. Попробуйте, например, сделать так: Map map = Map.of("a", 10, "b", 20); map.put("c", 30); и охренейте от того, что вам прилетит Exception в Runtime. При том что вся эта лабуда прекрасно скомпилируется. Когда-то налип с этим делом по незнанию, пришлось в довольно большом пакете искать где я делал Map.of и переделывать на new HashMap.
@dmytromarchuk3023
@dmytromarchuk3023 4 жыл бұрын
Ставлю лайк відразу! "Виріс" на відосах Сергія про Design Patterns, Clean Code, etc
@ArtemGaleev-w4t
@ArtemGaleev-w4t 4 жыл бұрын
Раз уж вы сказали про наследование и попросили предложить что после СОЛИД рассказать, то предлагаю достаточно холиварную тему: наследование vs композиция. Заранее прошу прощения если вы эту тему уже поднимали
@SergeyNemchinskiy
@SergeyNemchinskiy 4 жыл бұрын
расскажу. Регулярно поднимаю эту тему на тренингах, надо и в отдельном видео рассказать
@kirilleremeev9707
@kirilleremeev9707 4 жыл бұрын
Ну так, а какое все таки решение в вопросе прямоугольник-квадрат? Все его приводят в пример, а решения ни кто не приводит! Получается два выбора: либо отказаться от наследования и следовать LSP... Но тогда будет нарушен другой же принцип - не дублируй код. Т.е. если у вас в базовом классе (например в прямоугольнике) двести методов и один из них не LSP, то что тогда?
@SergeyNemchinskiy
@SergeyNemchinskiy 4 жыл бұрын
наследовать прямоугольник от квадрата
@vanweyden
@vanweyden 4 жыл бұрын
Вообще квадрат не нужен)
@kirilleremeev9707
@kirilleremeev9707 4 жыл бұрын
@@SergeyNemchinskiy Квадрат это частный случай от прямоугольника (по крайней мере я лучше выкину принципы solid чем так поступлю). И функции setWidth и setHight нужно будет переписывать для прямоугольника, что опять же нарушает - дочерний класс переопределяет функциональность базового. Теперь новый side эффект - я ожидаю что стороны равны для квадарта (базового класса), а для прямоугольника - не равны. Проблема с функцией площади теперь выливается аналогичную проблему с функцией с периметром...
@Hellsome_Rider
@Hellsome_Rider 4 жыл бұрын
Это вообще из другой оперы, но я бы послушал про DDD domain driven design, что такое bounded context и как его правильно определять
@НикитаГлухов-п5ю
@НикитаГлухов-п5ю 4 жыл бұрын
Про наследование в конце дичь. Это прямой способ насрать себе в руки, создав сильнейшую связь между 2 классами. Проблему хрупкого базового класса никто не отменял. Composition over inheritance годами проверенный принцип, приведший к тому, что в языках последнего десятилетия (Golang) наследование выкидывают нафиг как концепцию. Как бы для эффективного переиспользования кода и достижения гибкости, достаточно интерфейсов (декларирующих контракт) и делегирования. Об этом в Effective Java Джошуа Блох пишет почти в самом начале.
@Telemahk
@Telemahk 3 жыл бұрын
По итогу - надо квадрат наследовать от прямоугольника или правильнее новый класс делать?
@eligolin9947
@eligolin9947 2 жыл бұрын
Я постараюсь привести пример проблематики соблюдения LSP принципа. Допустим у вас есть класс Human. У этого класса есть такие поля как; пол, возраст, рост, вес и тд.. Теперь представим что у нас есть класс Student. В этом классе добавляется название колледжа, номер курса, и специальность. Согласитесь что наследование студента от человека выглядит довольно безобидным и естественным. Ещё представим что в текущей версии кода есть метода которая вычисляет налог "социального страхования" для любого человека, и основана она только на возрасте человека. Понятно что в текущей версии кода, LSP принцип соблюдается для этой методы. А теперь представим что в будущем закон изменился и теперь студенты освобождены от оплаты налогов. С этого момента LSP принцип нарушен. проблема здесь на самом деле глубокая и заключается она в том что если B наследует от A, то требуется доказать математически что B является A. Дело в том что наследование Student от Human основывается на нашем естественном восприятии но в то же время плохо определено в математическом смысле. Вся сложность LSP является в том что наследование должно быть корректно не только в текущей версии кода, но и во всех (неизвестно на сегодня) будущих его версиях. Когда домейн классов находится в математической плоскости нам легче создать корректное наследование, но к сожалению это лишь малая часть всех проблем которыми мы занимаемся. Отсюда вытекает общий антипаттерн наследования.
@OlegGoodsoul
@OlegGoodsoul Жыл бұрын
Цілком погоджуюсь.
@undefined-n5v
@undefined-n5v 4 жыл бұрын
Сложно представить сценарий, когда при наследовании захочется переопределить какой-то метод базового класса чтобы он кидал NotIImplementedException, у кого был грешок, расскажите, что вас к этому привело?
@screamer8932
@screamer8932 2 жыл бұрын
Насколько я понял из поиска по гуглу. Тут весь пример завязан на Mock об'екте который по определению является пустышкой с методами заглушками. И если оставлять родительськую реализацию то нарушается смысл. Это как я понял. Если кто поправит буду рад
@tislambek
@tislambek 4 жыл бұрын
Можете расказать в следующем видео о проекты спринга
@makskors5002
@makskors5002 3 жыл бұрын
TDD очень интересно!
@wcode404
@wcode404 4 жыл бұрын
Квадрат выглядит как частный случай прямоугольника, так же как квадрат частный случай ромба или прямоугольник частный случай параллелограмма. На самом деле это всё фигуры и на одна из них не является дочерней по отношению к другой. Я писал об этом здесь: github.com/peter-gribanov/clean-code-php
@ReAgent003
@ReAgent003 4 жыл бұрын
11:02 так всё-таки какая ситуация лучше: перезаписать сторону или выбросить исключение? Как в данной ситуации действовать?
@andriiv7033
@andriiv7033 3 жыл бұрын
Никак. Только нарушать принцип OCP - управляющий код должен знать про два класса. Так написано у Мартина.
@JackFastGame
@JackFastGame 4 жыл бұрын
А что является клиентским кодом в Unity?
@antonkuranov
@antonkuranov 4 жыл бұрын
Принцип подстановки безусловно правильный, но реально работает лишь в отдельных ограниченных случаях. Начиная с определенного момента ваши абстракции начинают течь, и решить эту проблему становится гораздо сложнее, нежели если бы LSP вообще не использовался бы с самого начала: приходится расширять интерфейс и патчить все зависимости. Пример протекающей абстракции -- это любой интерфейс драйвера, который возвращает "driver capabilities".
@FrolovDaniil
@FrolovDaniil 4 жыл бұрын
Можно ещё про DRY, KISS, YAGNI
@SergeyNemchinskiy
@SergeyNemchinskiy 4 жыл бұрын
уже в планах
@FrolovDaniil
@FrolovDaniil 4 жыл бұрын
@@SergeyNemchinskiy Смотрю Вас из интереса уже наверное больше года, когда ещё начинал знакомиться с Java, сейчас вообще Python разработчик. Так вот качество видосов стало на порядок выше, видно, что развиваетесь. Так держать! Пожалуйста, подумайте по поводу маркерной доски, а то эти шуршания маркера по бумаге некоторых приводят в ужас (меня например).
@OlezhikChanal
@OlezhikChanal 4 жыл бұрын
Раз пошла такая пьянка. Серию про паттерны жду. Буду смотреть и советовать всем.
@SergeyNemchinskiy
@SergeyNemchinskiy 4 жыл бұрын
Ну про паттерны я уже снял. Вот она foxminded.com.ua/grasp-gof-design-patterns-advanced-on-line-course/
@r2d2925
@r2d2925 4 жыл бұрын
Мне интересно о тдд, но желательно пример живой разработки, так как писать тесты которые падают перед программированиям меня немного смущает.
@denisbielishev
@denisbielishev 4 жыл бұрын
Будет интересно услышать про GRASP и GoF
@alexei3366
@alexei3366 4 жыл бұрын
можете записать видео почему метод не должен возвращать null?
@radikovichkz2470
@radikovichkz2470 Жыл бұрын
А как на счет предпочитайте композицию наследованию?
@Lammax2012
@Lammax2012 4 жыл бұрын
TDD - очень интересно!!!
@gomer3894
@gomer3894 4 жыл бұрын
спасибо за видео!
@vincentvega1903
@vincentvega1903 4 жыл бұрын
I love you Nemchinsy so bad
@haqon2k
@haqon2k 4 жыл бұрын
В целом, неплохо рассказано, конечно. Я бы добавил в формат реальный кейс с кодом, примеров нарушений принципа в системных библиотеках (в той же Java нарушения лисков достаточно). Ещё не затронута тема ковариантности и контравариантности. Иии что значит нельзя возвращать null из методов? Это абсолютно нормально и даже нужно. Естественно, с документацией. Или когда ваш язык поддерживает нормальные nullable типы - null как отдельный тип данных.
@SergeyNemchinskiy
@SergeyNemchinskiy 4 жыл бұрын
нельзя возвращать null, это очевидно
@haqon2k
@haqon2k 4 жыл бұрын
​@@SergeyNemchinskiy Если вы имеете ввиду принцип Лисков, то и тут не соглашусь, всё зависит от контракта, предоставляемым базовым типом, если там подразумевается null, то можно. А если вы имеете ввиду вообще, в целом, то и тут неясна ваша идея. В некоторых случаях, действительно, есть лучше способ показать отсутствие данных. Например, в .NET - try pattern, когда возвращаемое значение - bool, показывающее результат операции, а данные возвращаются через ссылку одним из аргументов. Также в .net ввели nullable reference types, с помощью которого явно видно, где тип может быть null, а где нет.
@0imax
@0imax 4 жыл бұрын
Я бы предпочёл получить из метода пустой список вместо null, прописанного где-нибудь в дебрях в документации.
@haqon2k
@haqon2k 4 жыл бұрын
0imax Касаемо чего то перечисляемого, да, тут лучшим способом будет вернуть что то пустое, но существующее. И то стоит понимать, что могут быть моменты, когда нужно уметь различать отсутсвие самого списка и отсутствие в списке вообще кого то. Но обычные объекты, возвращаемые аля GetUserById, GetActualPlayerConnection и прочее - здесь лучшим способом заявить об отсутствии чего то только null, никаких throw
@0imax
@0imax 4 жыл бұрын
@@haqon2k а по мне так лучше кинуть нормальный exception и обработать его, чем возвращать null: Во-первых, правильно названный exception сразу даёт информацию о причине "поломки". А во-вторых, таким образом можно обработать ситуацию, когда причин отсутствия объекта может быть несколько, как с тем же connection.
Quando eu quero Sushi (sem desperdiçar) 🍣
00:26
Los Wagners
Рет қаралды 15 МЛН
Cat mode and a glass of water #family #humor #fun
00:22
Kotiki_Z
Рет қаралды 42 МЛН
coco在求救? #小丑 #天使 #shorts
00:29
好人小丑
Рет қаралды 120 МЛН
Почему нельзя возвращать NULL?
22:11
Sergey Nemchinskiy
Рет қаралды 117 М.
Принцип хорошего кода YAGNI ("You aren't gonna need it")
15:08
Принцип подстановки Лисков. SOLID для React
15:26
Михаил Непомнящий
Рет қаралды 13 М.
THE MOST FREQUENT MISCONCEPTIONS ABOUT OOP
19:37
ExtremeCode
Рет қаралды 562 М.