Lol, I'm studying design patterns after studying some functional programming and this is what I just realised. Strategy pattern == first class functions, visitor pattern == pattern matching!
Reasons I find these react type videos to be among the top best things to be happening to my career: 1) I find out amazingly informative tech RUclipsrs from the original videos 2) Primes constant interjections is basically 100x-ing the information 3) He’s so f*in funny. 4) This kind of gold only seems to flow from highly technical senior engineers in the creamy layer of companies Thanks prime, continue to make these !
No offense to prime, and he's about as good an engineer as it gets. People that write business apps adopt the OO things because its useful in that context. engineering for large scale isn't the right context. I know people making $300k a year building salesforce apps, and $300k a year building large scale streaming systems. Context.
That's how the whole "patterns" thing started... a bunch of 4 guys were just trying to codify good ideas they'd seen so they could share them around. Things then... well. Things got a little out of hand. MyClassFactory myClassFactory = new MyClassFactory(); MyClass myClass = myClassFactory.create(); BWA HA HAH AHA HAHAH AHAH AHAAA
I remember being asked which patter is a smart pointer in c++. I listed 3, before they told me it was a proxy and i was like yeah, i guess, but who cares?
8:08 Well, as an avid Python programmer, I can vouch that while using decorators feels pretty nice (most of the time), writing them is an absolute mind numbing experience.
@@ShadowKestrel I'm doing the proc_macro workshop and doing the derive(Builder) macro as my first 1 since yesterday. It's all fun and games, till I got to the optional-field part. I shudder at the realization that there are so many way to make a struct field optional and I would need to check all of them. I'm just starting out with it so there are without a doubt tons of pattern and tools to help with these kind of things that I don't know of. Although do think it's probably a very useful tool to have in my arsenal and way more powerful than macro_rules.
how a decorator works: you write a function that takes in a function and gives back the "same" function plus you can do code on the side def print_instead(func): return print will give you a decorate that replaces the decorated function with print :D
I too love the strategy pattern. Higher order functions are the most beautiful way to do the strategy pattern. Functional programming is just the strategy pattern all the way down. I love the strategy pattern btw.
i was a bit confused at the begining of your message - does he rly love the strategy pattern, but at the end I put aside my doubts - you DO LOVE STRATEGY PATTERN!
It would be cool to go through examples of patterns in rust. I haven't looked at the code for serde but it seems to use interesting patterns such as adapters to support different serialization formats.
RUclips REALLY wanted me to watch this, popping up on my end screens and recommended list for like 2 weeks now. I finally watched it..... sadly I'm not far enough on my coding journey to have learned anything or understood what was going on other than the great Prime jokes🤣
Builder is my favourite. You can omit and add whatever you want, the depth and detail that you ask of the struct is (USUALLY) directly paired with the amount of chaining you do. You can go with defaults, specify a few parameters, add more if you need to, or you can do the equivalent of in-place initialisation (which although it may not be as performant unorganized unoptimised, compilers might recognise the simplicity of the chained functions and would be able to evaluate the code into an object literal).
I use the builder pattern for all my JS Web Components. I am surprised people are surprised at returning this. It works nicely with a functional style, too, with a builder class whose final commit returns a DOM node and other methods return this. The web component then basically builds as one function call that when returned returns the ready component. You may get away with using not a single if else for a full and fully parametrised component with ES6 if returning this from each build method except commit. Listeners and any other features are just properties inserted with the builder as each sub component is described by joining build calls with the dot notation. The build tree will thus extrapolate to any complexity, as a component can always comprise any of the builds, which can comprise any builds, and so on, till the call stack returns to the top level and a full HTML component with any child components is returned, and is inserted in the DOM with the final commit. In theory, the whole web page could be built thusly. Prettier than some React stuff I have seen.
the idea of patterns is great. how people found patterns and tried to put everything in a pattern shaped hole was not. specially when "patterns" became some sort of bizarre synonym of OOP
Some of the patterns are so ubiquitous that you almost don't need to know them. Like I don't need to know that air is called air because I just breathe it naturally
I'm just about to suggest in my project to start using a Factory for a single reason. We have 5 different classes that create the same object using a builder BUT it isn't clear how many fields of that class have to be set up(spoiler all of its parameters so always the complete builder). To isolate in a single place how the creation of that object has to be, enforce validations and ensure one of the fields that is a map is always created correctly depending on 1 of those 5 usecases. If not we will snowball into having distribuited the creation logic across multiple places
And *that* is what factories are made for. To unify and hide away complexity from the developer who shouldn't have to worry about that stuff, and you still need to have some unified way to create everything. Oh, and I dislike factories (honestly not too keen of builders either) so I have every reason to absolutely shit on them. It's just that in some cases they actually do make sense. They are simply abused to infinity and beyond.
@@CottidaeSEA i feel like this how it goes for a lot of patterns. We hate them because of how they get misused, but by god they are very useful in their actual niche.
don't use a builder or factory if instantiation params are wonky/unknown. you should consider writing a context object. then any methods/classes that have the wonky/painful params pull from a single source. you'll see this a lot in JS when working with large libraries. There will usually be a single settings object which describes how the library should behave. The key to keeping settings objects maintainable is making them immutable. Once you allow mutations to settings, you are no longer passing around context, you are passing around state. immutable context is predictable state is not.
I usually think about the Facade pattern in terms of delineation between groups of developers, or major functions in an app... But one could just consider it a part of UX (or DX) → You make stuff easier to communicate, understand, and use. I wouldn't think of it normally as an endpoint though - unless you were explaining the endpoint with docs in a "This is all you need to know to do this" sense. But I think that perspective comes from seeing Facades as typically being a class with clear features, and not just any point that simplifies interactions between one system and other systems.
Peak of OO culture: Implementing Strategy and Visitor patterns (I still can't implement the visitor pattern without looking it up). Peak of FP culture: Monad transformers
@@wlockuz4467 SIMPLIFIED: it's a "wrapper", which takes closures/functions/lambdas which modifies the internal data (aka map function) It also needs a "pure/unit/wrap" function and also a bind/flatmap function. *Example without monads:* let person = Person { name: "John" }; function print_name(person: Person) { if person != null { if person.name != null { print(person.name) } } } print_name(person); *Example with a monad (Option):* let person = Person { name: "John" }; function print_name(person: Option) { person.map(p -> p.name).map(name -> print(name)); } print_name(person);
@@wlockuz4467 Instead of getting the data and applying your code to it (like a loop), you pass your code to the monad an IT will apply your code to the data. You just don't exactly know if, how or when cuz abstraction, which is the entire point. For example, in an Option/Maybe/Try monad your code will only be applied to the data if it exists.
outside of Monad Transformers you have MTL, Lenses and for many types of programs instead of using MTL many people use Freer Monads/Extensible Effects. Also arrows
Reactive programming works a lot with Observer, Iterator and Strategy patterns. For some historical insight, it's called RX because Microsoft did a lot of work with something called Reactive eXtensions, hence it evolved to RXJava.
I think the Singleton pattern as just a special case of "execution contexts" where there is only one global context. It can simplify code by removing the need to explicitly pass down dependencies into the callstack. However, as with all static globals, it creates global coupling. staying with the python example from the video I would always rather use a construct as below instead of a singleton: ```python with some_context_of_type_y(): with some_context_of_type_x(): foo() def foo(): y = get_y_object() x = get_x_object() ``` IMHO it gives better composability on the outside scope and has the same advantages as singletons on the inside scope.
Yea there are very few scenarios to actually use this pattern once you have DI, in fact it only interferes with/breaks DI. The only two examples I can think where I would actually use it are A) a tiny app where I'm not going to include an IoC library or B) In the very initial boot of your app when the IoC container/AppContext is either half constructed or has not finished initializing and is in an unsafe/incomplete state, yet you still need access to critical components like a preInit logger. Cause if AppContext fails to initialize... and that's where you get your logger from... and you want to log the error... and it IoC fails to initialize.... so you want to log it... but it has the..... wait.. where was I? where am i? WHO ARE YOU PEOPLE???? Can someone call my mom, im scared
I used to dislike decorators, but I'm currently working in C# (which doesn't have them) and I keep running into instances where it would be really nice... I have a class that makes API calls asynchronously, but I need to limit the number of async calls being made at a time. So, I need to use a semaphore to force threads to await if all the other threads have used up the semaphore. Would be super nice to just use a decorator at the top of every method that makes an API call so that it wraps the execution with an await to the semaphore, but not an option for vanilla C#. I ended up creating another class called RequestExecutor that has methods that take lambda functions and do this instead, but it feels less clean.
@@banatibor83 If you think decorators are bad, what about the abstractfactory patterns here. They are used for black-box aspect oriented programming to simplify your code, and are in JavaScript as well.
Observer, Iterator and Strategy are great. Singleton is ok if you're careful with state (Rx singleton Observable for instance). Adaptor and Facade are unavoidable, but I hate how coupled they can get. Factory and Builder are meh - I feel they're needed sometimes but they get overused. ps. lol at Primer reacting to Python OOP in real time
The idea of a factory is to provide only one interface create so client class could use it to create object of given interface. Like when you need some default value (like in data classes) when constructing object.
Builder and factory are just semantic variations on a constructor. Builder just turns parameter assignments into individual functions that can be chained together.
Thanks for going over this. I feel like I must be missing something because I’ve watched this and Fireship’s video on it and feel like this doesn’t actually help me think about the way data can be structured or organized. I’m a high school teacher, so working on learning to relay the lessons to students in my CS classes. Thanks Prime. 😊
The main idea is to not reinvent the wheel every time a similar problem occurs. Back when the book came out, the idea was that software engineering was a relatively new discipline in comparison to e.g. engineering or architecture. So they started to look for standard solutions in patterns at the design or implementation stage of a project
@@spittylama Thank you. I do understand that’s the *purpose*; I’m saying I can tell I still have yet to understand the content. I have been working on building projects with data, but I’m not sure I see the patterns, yet. I hope that makes sense.
Ngl... sometimes listening to the definitions of these named patterns,I feel like the next one is gonna be "and the Typing Pattern is where you press keys on the keyboard to put code in a computer!" "The Press Play Pattern is where you have to click a button to make your code run after it's written!"
This video was good. Why? We learnt something, unlike some videos where you just rant or do irrelevant optimisations, in this I actually learnt something which can get someone hired.
Common confusion: Observer is not Pub-Sub. Observers listen directly to their Observables/Subjects, so then each Observable/Subject must keep references to all of its Observers and become bound to their lifetimes. Publish-Subscribe adds one more level of indirection by having publishers and subscribers both depend directly upon a Channel between them; therefore they are tied to the lifetime of the channel rather than each other. Subscribers only become aware of publishers when notified with a message that provides the reference to the publisher if necessary.
Great comment at 15:48 . From a functional perspective, the "interface" you are crying out for, is just a function. However I don't think it's worth a full fedora-tip here, because as competent engineers we need to be prepared for any situation we may walk into. Such as George Clooney walked into in From Dusk Till Dawn.
Saw the original before, generally good comments as far as my experience with these patterns go. One critique I have is your statement that "Everything is a facade, becaus you hide information" with something something private. Facade does obfuscate variables and functions from the programmer, but they are still accessible from the outside. Basically, Facade (at least as stated in the GangOf4) is an interface to a functionality that says "Hey, 99% of the time, you will want to use me by calling these couple of functions here. If you need more custom behaviour and you REALLY know what you are doing, you can also tweak all my other variables and use these other functions". So Facade gives you a set of convenient functions to use it with, but also allows for access to everything else about the interfaved functionality, if you want to.
The only reason to use a factory is to hide the complexity of object creation. If a design pattern does not result in simpler code, then you are using the wrong design pattern.
Primogen really souds like a young Rick, from Rick and Morty. I dont have time for making this but just close your eyes and imagine it. Add some burps and voila. I think its the rhythm.
Close your eyes and listen and you half expect Prime to throw out a random "Morty" during his speeches. Y'ALL KNOW I'M RIGHT (seriously though if Prime wasn't a software dev he could probably take over for Rick's VA)
As a netflix btw guy Prime could/should have really said a bit more about the observer pattern. ReactiveX was a live changer for me, and that netflix talk about it sold it to me, back then.
I am convinced that the US has a different Java than the rest of us. The only place I know of that has abstract factories are frameworks like Spring. Because Spring can make a case for "I need to have a way of producing objects of whatever type specified", as well as "and I need to allow more than one way of doing it". And that's just because of the very generic way the IoC-container works.
Design patterns in functional programming: strategy pattern: functions builder pattern: functions observer pattern: functions adapter pattern: believe it or not also functions
It is only useful when you need to control what actually gets instantiated or how it gets configured during or after constructing. In all other cases, just doing a "new" of the proper class is both simpler and easier to follow. So if there is this need, sure, go for it. But I will never consider it a pattern to default to.
I did use a factory on my own. Needed to make a bot object that needed to load a few things from files assorted with a user Id. It was actually a ton of fun
As a Java developer I stay away from factories if I can. But yes, I've had my share of Complicators on teams. And yes, (fluent) builders to create immutable instances ❤
Developer-Experience is the missing pattern category. 17:55 Facade is a DX pattern for every clean http endpoint , convenience functions , the porcelain over the plumbing.
9:50 "shared mutable state is the bane to all programming and the key to super awesome amazing performance and it is the scariest thing in the universe...so...careful."
It's funny how the borrow from the French language lost the cédille, but kept the pronounciation : it's façade in French, where ç is pronounced like double s
Builder isn't "better" than factory, they are just suited to different circumstances - builder for flexibility; factory for simplicity. E.g. a library class uses builders in order to provide maximal functionality, but an app using that library wraps this in a factory to provide a simpler interface for the small number of specific uses its needed for in that app.
I just tend to find that factories are not what I want, yet. That's what I find myself doing. But a builder almost universally is what I want. Oftentimes you can even interchange them and I find the results almost always better with builder and a good setup defaults
12:50 i had the similar experience. Microsoft tried to hire me with a 40% cut of my city salary. They said oh but we do value you, we intend to hire you out to clients for $5000 a day. This was in 2000 when $5000 was still quite an impressive amount.
I usually prefer default in Rust over the builder pattern. I do emit part of this is because it's a lot easier to implement. I don't normally mind using the builder pattern but I hate programming it. Also I feel like it doesn't normally have much of a benefit so I don't know if it's worth implementing in most cases.
2:48, they just gave a name for a thing that has been made in C for decades: struct CheeseBurger { const char *ing_1, *ing_2, *ing_3 }; #define createCheeseBurger CheeseBurger { "bun", "cheese", "beef-patty" } And then call it by: CheeseBurger a_cheese_burger = createCheeseBurger; 5:20, this is a "lesser version" of C++ for 2 reasons: a) If 1 just wants to customize the initialization, C/C++ way is much better, as exposed above, due to compact syntax. b) If the goal is to customize it along the way, not only on initialization, it should has a control of which f() is allowed do that - _and it seems that only C++ has this feature_ . Otherwise, all sorts of bugs can come from it, because everything will be public, which is the worst nightmare, depending on the project complexity. 10:57, why use an interface for that? Couldn't it be just a class? 16:00, again, why an interface? This thing is awkwardly slow! 21:00, and that's why f() programming < OO: the 1st lacks good facades. In C, you can let that memory management "encapsulated" away in a different file. But there are issues: a) Each file like this will be dedicated to each object, which hurts scalability. b) If only 1 generic file like this is created, then the user will has to keep traveling alongside with that array, which is exposed to dumb mistakes for long unnecessary time. It's still possible to create several macros, trying to hide that array exposure, but it's still a precary solution.
The implementation of the linkedlist iterator in the video is actually faulty. If you would create two iterators they would actually both modify the object.
Singletons are fine if they contain no mutable state. At my work we use singleton lifetime DI extensively because all state is passed through arguments and returns. Classes become modules and interfaces make those modules loosely coupled. It's almost like functional programming but with a nicer syntax.
Why not just use a static class then? Genuinely curious the difference between a singleton and a static class, I'm sure they are drastically different I just don't know how.
@@grantpeterson2524 In javascript and python there is a memoize (called lru_cache) method that caches calls for functions, the memoization of a no argument function is called the singleton. Static classes run code at the start of your program, while singletons are initialized the first time they are used.
On the last point of the facade pattern, NeetCode says "you don't want to think about memory allocation" and here I'm thinking "it isn't difficult at all, there is this thing called realloc." The only reason why it sucks in java is because java does precisely this, remove any need for you to think about memory allocation. Realloc is a better abstraction than java.
Holy shit, theprimeagen reacted to me! 🤯 Love it
Thank you for 16:42 you damn comedic genius
You've made it! Keep that neet code (and those neet takes) coming. 🏆🥳
He just got you a new sub
Man, the Java joke was fantastic!
Whew, that one went your way... this time. :)
Interestingly there are also functional programming patterns. I've got the full list here:
1. Functions
I have been studying design pattern for some exams and this is just GOLD!
Thank you for the laugh!😂
That is all you need
Lol, I'm studying design patterns after studying some functional programming and this is what I just realised. Strategy pattern == first class functions, visitor pattern == pattern matching!
True, Strategy pattern = higher order function, Iterator pattern = functor
Completely ignoring how functions can get other functions as an input, or even self-invoke them.
Reasons I find these react type videos to be among the top best things to be happening to my career:
1) I find out amazingly informative tech RUclipsrs from the original videos
2) Primes constant interjections is basically 100x-ing the information
3) He’s so f*in funny.
4) This kind of gold only seems to flow from highly technical senior engineers in the creamy layer of companies
Thanks prime, continue to make these !
Couldn't agree more !
No offense to prime, and he's about as good an engineer as it gets. People that write business apps adopt the OO things because its useful in that context. engineering for large scale isn't the right context. I know people making $300k a year building salesforce apps, and $300k a year building large scale streaming systems. Context.
Constructive criticism: "Creamy Layer of companies" makes me uncomfortable 🥴
Reason # 5) You have brain damage.
Agreed
I love how some of the patterns are "wtf is that monstrosity" while others are "oh, that's a pattern? I thought that's basic doing things".
That's how the whole "patterns" thing started... a bunch of 4 guys were just trying to codify good ideas they'd seen so they could share them around. Things then... well. Things got a little out of hand.
MyClassFactory myClassFactory = new MyClassFactory();
MyClass myClass = myClassFactory.create();
BWA HA HAH AHA HAHAH AHAH AHAAA
I remember being asked which patter is a smart pointer in c++. I listed 3, before they told me it was a proxy and i was like yeah, i guess, but who cares?
Exactly, for example Singleton is simply a fancy name for a global variable!
2:48 - factory pattern
4:38 - builder pattern
6:36 - -singleton- arch user pattern
10:03 - observer pattern
11:40 - iterator pattern
14:21 - strategy pattern
16:09 - adapter pattern
17:56 - facade pattern
more like arch-user-ton patter :)
Arch user pattern aka game developer pattern.
"strategy pattern" aka function pointer
@@peaked2258 UGH... no.
Façade
So many religions in software nowadays
It's kinda weird ngl
Even IRL religions don't make sense, let alone software religions
Yeah, only religion that's valid it HolyC and TempleOS
Alahuakbar
@@BboyKeny long live the schizo god!!!!
(he works at Netflix btw)
I wonder where he works
Just realized 1994 is 30 years ago
tf?
i thought 30 years ago its 1970
The pattern of choice at my job: The Spaghetti Pattern
8:08 Well, as an avid Python programmer, I can vouch that while using decorators feels pretty nice (most of the time), writing them is an absolute mind numbing experience.
same goes for rust macros: I love when they are provided but now I am writing them my sanity is simply not here any more
@@ShadowKestrel I'm doing the proc_macro workshop and doing the derive(Builder) macro as my first 1 since yesterday.
It's all fun and games, till I got to the optional-field part. I shudder at the realization that there are so many way to make a struct field optional and I would need to check all of them.
I'm just starting out with it so there are without a doubt tons of pattern and tools to help with these kind of things that I don't know of.
Although do think it's probably a very useful tool to have in my arsenal and way more powerful than macro_rules.
Me: "How does decorator work behind the scenes?"
Senior dev: "It just works"
@@ThatOpinionIsWrong with a smile on its face, that's how!
how a decorator works:
you write a function that takes in a function and gives back the "same" function
plus you can do code on the side
def print_instead(func):
return print
will give you a decorate that replaces the decorated function with print :D
I too love the strategy pattern. Higher order functions are the most beautiful way to do the strategy pattern. Functional programming is just the strategy pattern all the way down. I love the strategy pattern btw.
Do you love the strategy pattern?
@@iambilbobaggins1884 I think he loves the strategy pattern... Can he confirm?
@@Gamester-vy1qp I, too, wish I knew how he felt about this, and where/if he works.
i was a bit confused at the begining of your message - does he rly love the strategy pattern, but at the end I put aside my doubts - you DO LOVE STRATEGY PATTERN!
Bro. I think you love the strategy pattern
are we going to just ignore the fact that he's using camelCase on Python?
@Chase Miller snake_case >>>>> everything else
@@tablettablete186 camel case is the only way
@@judewaide8328 oBjEcTivALy WRONG!!! 😤
(I wanna where this goes 😅😅😅)
@@tablettablete186 lol
@@tablettablete186 snake_case gang
> Buys patterns book
> Looks inside
> Interfaces
> 😐
It would be cool to go through examples of patterns in rust. I haven't looked at the code for serde but it seems to use interesting patterns such as adapters to support different serialization formats.
I'm gonna take a shot every time you mention you work at Netflix
should i call an ambulance?
RUclips REALLY wanted me to watch this, popping up on my end screens and recommended list for like 2 weeks now. I finally watched it..... sadly I'm not far enough on my coding journey to have learned anything or understood what was going on other than the great Prime jokes🤣
Singletons are really nice for hardware abstractions in embedded systems. For example I only want one instance of my keyboard structure, not 2 or 20.
I actually want 20, one for japanese , English and macro all in 1 keyboard
Builder is my favourite. You can omit and add whatever you want, the depth and detail that you ask of the struct is (USUALLY) directly paired with the amount of chaining you do. You can go with defaults, specify a few parameters, add more if you need to, or you can do the equivalent of in-place initialisation (which although it may not be as performant unorganized unoptimised, compilers might recognise the simplicity of the chained functions and would be able to evaluate the code into an object literal).
I use the builder pattern for all my JS Web Components. I am surprised people are surprised at returning this. It works nicely with a functional style, too, with a builder class whose final commit returns a DOM node and other methods return this. The web component then basically builds as one function call that when returned returns the ready component.
You may get away with using not a single if else for a full and fully parametrised component with ES6 if returning this from each build method except commit. Listeners and any other features are just properties inserted with the builder as each sub component is described by joining build calls with the dot notation. The build tree will thus extrapolate to any complexity, as a component can always comprise any of the builds, which can comprise any builds, and so on, till the call stack returns to the top level and a full HTML component with any child components is returned, and is inserted in the DOM with the final commit. In theory, the whole web page could be built thusly.
Prettier than some React stuff I have seen.
the idea of patterns is great.
how people found patterns and tried to put everything in a pattern shaped hole was not.
specially when "patterns" became some sort of bizarre synonym of OOP
When someone confused patterns with OOP and confused OOP with classes it is quite difficult to take them seriously
@@avwie132i don't program, they come off pretentious at the least 😢
Why do you think that happened?
My suspicion is that most Modeling languages focus on OOP.
Video: Screw too small for the hole.
Primeagen: We've all felt this...
Me: I thought the idea was you couldn't feel it.
You know what you should feel by knowing what you don’t feel.
Some of the patterns are so ubiquitous that you almost don't need to know them. Like I don't need to know that air is called air because I just breathe it naturally
I'm just about to suggest in my project to start using a Factory for a single reason. We have 5 different classes that create the same object using a builder BUT it isn't clear how many fields of that class have to be set up(spoiler all of its parameters so always the complete builder). To isolate in a single place how the creation of that object has to be, enforce validations and ensure one of the fields that is a map is always created correctly depending on 1 of those 5 usecases. If not we will snowball into having distribuited the creation logic across multiple places
And *that* is what factories are made for. To unify and hide away complexity from the developer who shouldn't have to worry about that stuff, and you still need to have some unified way to create everything.
Oh, and I dislike factories (honestly not too keen of builders either) so I have every reason to absolutely shit on them. It's just that in some cases they actually do make sense. They are simply abused to infinity and beyond.
@@CottidaeSEA i feel like this how it goes for a lot of patterns. We hate them because of how they get misused, but by god they are very useful in their actual niche.
don't use a builder or factory if instantiation params are wonky/unknown. you should consider writing a context object. then any methods/classes that have the wonky/painful params pull from a single source. you'll see this a lot in JS when working with large libraries. There will usually be a single settings object which describes how the library should behave.
The key to keeping settings objects maintainable is making them immutable. Once you allow mutations to settings, you are no longer passing around context, you are passing around state. immutable context is predictable state is not.
@@NihongoWakannai Yeah, I feel like that also translates to OOP getting a lot of undeserved hate.
@@wadecodez Doesn't that pattern also use a factory? I've never seen it without a factory at least.
Python decorators are the best. They are the easiest form I saw for metaprogramming. It is really easy to read and understand the logic behind them
I usually think about the Facade pattern in terms of delineation between groups of developers, or major functions in an app... But one could just consider it a part of UX (or DX) → You make stuff easier to communicate, understand, and use.
I wouldn't think of it normally as an endpoint though - unless you were explaining the endpoint with docs in a "This is all you need to know to do this" sense.
But I think that perspective comes from seeing Facades as typically being a class with clear features, and not just any point that simplifies interactions between one system and other systems.
Peak of OO culture: Implementing Strategy and Visitor patterns (I still can't implement the visitor pattern without looking it up).
Peak of FP culture: Monad transformers
I still don't know wtf is a monad.
@@wlockuz4467 SIMPLIFIED: it's a "wrapper", which takes closures/functions/lambdas which modifies the internal data (aka map function)
It also needs a "pure/unit/wrap" function and also a bind/flatmap function.
*Example without monads:*
let person = Person { name: "John" };
function print_name(person: Person) {
if person != null {
if person.name != null {
print(person.name)
}
}
}
print_name(person);
*Example with a monad (Option):*
let person = Person { name: "John" };
function print_name(person: Option) {
person.map(p -> p.name).map(name -> print(name));
}
print_name(person);
@@wlockuz4467 Instead of getting the data and applying your code to it (like a loop), you pass your code to the monad an IT will apply your code to the data. You just don't exactly know if, how or when cuz abstraction, which is the entire point. For example, in an Option/Maybe/Try monad your code will only be applied to the data if it exists.
outside of Monad Transformers you have MTL, Lenses and for many types of programs instead of using MTL many people use Freer Monads/Extensible Effects. Also arrows
Reactive programming works a lot with Observer, Iterator and Strategy patterns. For some historical insight, it's called RX because Microsoft did a lot of work with something called Reactive eXtensions, hence it evolved to RXJava.
I think the Singleton pattern as just a special case of "execution contexts" where there is only one global context.
It can simplify code by removing the need to explicitly pass down dependencies into the callstack.
However, as with all static globals, it creates global coupling.
staying with the python example from the video I would always rather use a construct as below instead of a singleton:
```python
with some_context_of_type_y():
with some_context_of_type_x():
foo()
def foo():
y = get_y_object()
x = get_x_object()
```
IMHO it gives better composability on the outside scope and has the same advantages as singletons on the inside scope.
Yea there are very few scenarios to actually use this pattern once you have DI, in fact it only interferes with/breaks DI. The only two examples I can think where I would actually use it are A) a tiny app where I'm not going to include an IoC library or B) In the very initial boot of your app when the IoC container/AppContext is either half constructed or has not finished initializing and is in an unsafe/incomplete state, yet you still need access to critical components like a preInit logger.
Cause if AppContext fails to initialize... and that's where you get your logger from... and you want to log the error... and it IoC fails to initialize.... so you want to log it... but it has the..... wait.. where was I? where am i? WHO ARE YOU PEOPLE???? Can someone call my mom, im scared
Decorators are one of my favorite things about Python. They're so fantastic.
One of the most useless language feature :) They have very limited use if you do not want to end up with totally unreadable code.
I used to dislike decorators, but I'm currently working in C# (which doesn't have them) and I keep running into instances where it would be really nice...
I have a class that makes API calls asynchronously, but I need to limit the number of async calls being made at a time. So, I need to use a semaphore to force threads to await if all the other threads have used up the semaphore. Would be super nice to just use a decorator at the top of every method that makes an API call so that it wraps the execution with an await to the semaphore, but not an option for vanilla C#. I ended up creating another class called RequestExecutor that has methods that take lambda functions and do this instead, but it feels less clean.
@@banatibor83 If you think decorators are bad, what about the abstractfactory patterns here. They are used for black-box aspect oriented programming to simplify your code, and are in JavaScript as well.
I don't know why but I crave more of these 💓💓
Observer, Iterator and Strategy are great. Singleton is ok if you're careful with state (Rx singleton Observable for instance). Adaptor and Facade are unavoidable, but I hate how coupled they can get. Factory and Builder are meh - I feel they're needed sometimes but they get overused. ps. lol at Primer reacting to Python OOP in real time
the arch user strategy pattern sounds interesting
The idea of a factory is to provide only one interface create so client class could use it to create object of given interface. Like when you need some default value (like in data classes) when constructing object.
You mean like a function 😅
Builder and factory are just semantic variations on a constructor. Builder just turns parameter assignments into individual functions that can be chained together.
This is exploding my brain - ive had such a hard time listening to other peoples coding videos. Not this, im fully engaged. Thank you!
16:30 I recently had a similar problem.
My solution was buying bigger screws.
i really didnt know that prime worked at netflix, im glad he mentioned it 10 times
Thanks for going over this. I feel like I must be missing something because I’ve watched this and Fireship’s video on it and feel like this doesn’t actually help me think about the way data can be structured or organized.
I’m a high school teacher, so working on learning to relay the lessons to students in my CS classes. Thanks Prime. 😊
The main idea is to not reinvent the wheel every time a similar problem occurs. Back when the book came out, the idea was that software engineering was a relatively new discipline in comparison to e.g. engineering or architecture. So they started to look for standard solutions in patterns at the design or implementation stage of a project
@@spittylama Thank you. I do understand that’s the *purpose*; I’m saying I can tell I still have yet to understand the content.
I have been working on building projects with data, but I’m not sure I see the patterns, yet. I hope that makes sense.
@@mchisolm0 patterns with data are even more recent! So no worries with that. The well established patterns often have a OO design approach.
@@spittylama do are you saying this is *not* about patterns with data? If so, I must have been really misunderstanding…
You could have a class called Food.
It can be composed of an array of various ingredients and include a name variable.
“Patterns” is just something to grab if you don’t have lambda, the ultimate abstraction
Ngl... sometimes listening to the definitions of these named patterns,I feel like the next one is gonna be "and the Typing Pattern is where you press keys on the keyboard to put code in a computer!" "The Press Play Pattern is where you have to click a button to make your code run after it's written!"
That casual "when I was at Google" also got me going "weird place to flex, bud" 😂
This video was good. Why? We learnt something, unlike some videos where you just rant or do irrelevant optimisations, in this I actually learnt something which can get someone hired.
Rant material is often a flash-forward to a problem or conflict that you just haven't run into YET -- artifacts of some clear, prior trauma.
Common confusion: Observer is not Pub-Sub. Observers listen directly to their Observables/Subjects, so then each Observable/Subject must keep references to all of its Observers and become bound to their lifetimes. Publish-Subscribe adds one more level of indirection by having publishers and subscribers both depend directly upon a Channel between them; therefore they are tied to the lifetime of the channel rather than each other. Subscribers only become aware of publishers when notified with a message that provides the reference to the publisher if necessary.
Great comment at 15:48 . From a functional perspective, the "interface" you are crying out for, is just a function.
However I don't think it's worth a full fedora-tip here, because as competent engineers we need to be prepared for any situation we may walk into. Such as George Clooney walked into in From Dusk Till Dawn.
Saw the original before, generally good comments as far as my experience with these patterns go.
One critique I have is your statement that "Everything is a facade, becaus you hide information" with something something private. Facade does obfuscate variables and functions from the programmer, but they are still accessible from the outside. Basically, Facade (at least as stated in the GangOf4) is an interface to a functionality that says "Hey, 99% of the time, you will want to use me by calling these couple of functions here. If you need more custom behaviour and you REALLY know what you are doing, you can also tweak all my other variables and use these other functions". So Facade gives you a set of convenient functions to use it with, but also allows for access to everything else about the interfaved functionality, if you want to.
I love how straight to the point reactions Prime Reacts have
I loved head first design patterns 🥰
The book is just so awesome. The way they teach design pattern by first introducing the problem and slowly building up the solution is great!
8:25 - What is this contraption??
16:18 - ngl this caught me off guard mid drink
The only reason to use a factory is to hide the complexity of object creation. If a design pattern does not result in simpler code, then you are using the wrong design pattern.
Primogen really souds like a young Rick, from Rick and Morty. I dont have time for making this but just close your eyes and imagine it. Add some burps and voila. I think its the rhythm.
Close your eyes and listen and you half expect Prime to throw out a random "Morty" during his speeches. Y'ALL KNOW I'M RIGHT (seriously though if Prime wasn't a software dev he could probably take over for Rick's VA)
As a netflix btw guy Prime could/should have really said a bit more about the observer pattern. ReactiveX was a live changer for me, and that netflix talk about it sold it to me, back then.
I am convinced that the US has a different Java than the rest of us.
The only place I know of that has abstract factories are frameworks like Spring.
Because Spring can make a case for "I need to have a way of producing objects of whatever type specified", as well as "and I need to allow more than one way of doing it".
And that's just because of the very generic way the IoC-container works.
Design patterns in functional programming:
strategy pattern: functions
builder pattern: functions
observer pattern: functions
adapter pattern: believe it or not also functions
Builder pattern is record syntax + functions, adapter is record of functions.
Functional programming uses functions instead of objects, this checks out.
I love the builder pattern, it's so useful. And the factory pattern is also super helpful
In Python keyword arguments and dictionaries tetrisgame(func1={arg1: 5}, object2={arg2: object3}) can replace builder pattern...
@@aoeu256can you elaborate?
It is only useful when you need to control what actually gets instantiated or how it gets configured during or after constructing.
In all other cases, just doing a "new" of the proper class is both simpler and easier to follow.
So if there is this need, sure, go for it.
But I will never consider it a pattern to default to.
I did use a factory on my own.
Needed to make a bot object that needed to load a few things from files assorted with a user Id.
It was actually a ton of fun
Dr.disrespect of programming
this channel is my favorite netflix advertisement
builder pattern with rust is good because of the TypeState Pattern
This guy is intense. But i think he works at Netflix right?
I first thought it was about PrimeReact UI Library.
Your videos are fun to watch; I've subscribed.
As a Java developer I stay away from factories if I can. But yes, I've had my share of Complicators on teams.
And yes, (fluent) builders to create immutable instances ❤
I hate the new keyword that forces you differentiate between factory methods and class constructors, well at least the IDE can fix it for you.
Haha! Creational patterns look to me like a solution to several problems all bundled together as OOPsies!
I had that book. I tore it apart before I trashed it so nobody else would be infected by it.
That opening was like a man confessing adultery to his wife.
Facade = Noise Adapter
Adapter = Interface Strategy
Strategy Pattern = interface.method(strategy) = noun.verb(how)
Adapter = noun2.verb(noun1.verb(how))
Facade = () => noun2.verb(noun1.verb(how))
love the design patterns book, great video!
Developer-Experience is the missing pattern category.
17:55 Facade is a DX pattern for every clean http endpoint , convenience functions , the porcelain over the plumbing.
9:50 "shared mutable state is the bane to all programming and the key to super awesome amazing performance and it is the scariest thing in the universe...so...careful."
Iterators in java: sugar
Iterators in Rust: honey
Iterators in python: 8 year old stevia packet
It's funny how the borrow from the French language lost the cédille, but kept the pronounciation : it's façade in French, where ç is pronounced like double s
Builder isn't "better" than factory, they are just suited to different circumstances - builder for flexibility; factory for simplicity. E.g. a library class uses builders in order to provide maximal functionality, but an app using that library wraps this in a factory to provide a simpler interface for the small number of specific uses its needed for in that app.
I just tend to find that factories are not what I want, yet. That's what I find myself doing. But a builder almost universally is what I want. Oftentimes you can even interchange them and I find the results almost always better with builder and a good setup defaults
I agree with Prime. I’ve never liked factories. They aren’t as flexible as builders.
17:45 Adapter pattern is just the application of a strategy pattern.
4:20; Composition over Inheritance
12:50 i had the similar experience. Microsoft tried to hire me with a 40% cut of my city salary. They said oh but we do value you, we intend to hire you out to clients for $5000 a day. This was in 2000 when $5000 was still quite an impressive amount.
"I work at Netflix btw"
has similar vibe as
"I use Arch btw"
Can you not say "okay google" while I'm driving? It pauses the video! :p
Laughed at the, "is this bash now!?" comment 😂
First video from this guy that I like
I usually prefer default in Rust over the builder pattern. I do emit part of this is because it's a lot easier to implement. I don't normally mind using the builder pattern but I hate programming it. Also I feel like it doesn't normally have much of a benefit so I don't know if it's worth implementing in most cases.
2:48, they just gave a name for a thing that has been made in C for decades:
struct CheeseBurger { const char *ing_1, *ing_2, *ing_3 };
#define createCheeseBurger CheeseBurger { "bun", "cheese", "beef-patty" }
And then call it by:
CheeseBurger a_cheese_burger = createCheeseBurger;
5:20, this is a "lesser version" of C++ for 2 reasons:
a) If 1 just wants to customize the initialization, C/C++ way is much better, as exposed above, due to compact syntax.
b) If the goal is to customize it along the way, not only on initialization, it should has a control of which f() is allowed do that - _and it seems that only C++ has this feature_ . Otherwise, all sorts of bugs can come from it, because everything will be public, which is the worst nightmare, depending on the project complexity.
10:57, why use an interface for that? Couldn't it be just a class? 16:00, again, why an interface? This thing is awkwardly slow!
21:00, and that's why f() programming < OO: the 1st lacks good facades. In C, you can let that memory management "encapsulated" away in a different file. But there are issues:
a) Each file like this will be dedicated to each object, which hurts scalability.
b) If only 1 generic file like this is created, then the user will has to keep traveling alongside with that array, which is exposed to dumb mistakes for long unnecessary time.
It's still possible to create several macros, trying to hide that array exposure, but it's still a precary solution.
we need more abstract builder factories producing abstract builder factories.
The implementation of the linkedlist iterator in the video is actually faulty. If you would create two iterators they would actually both modify the object.
14:00 I don't think you're supposed to raise exceptions on regular functionality (reaching the end of the list)
Singletons are fine if they contain no mutable state.
At my work we use singleton lifetime DI extensively because all state is passed through arguments and returns. Classes become modules and interfaces make those modules loosely coupled.
It's almost like functional programming but with a nicer syntax.
Why not just use a static class then? Genuinely curious the difference between a singleton and a static class, I'm sure they are drastically different I just don't know how.
@@grantpeterson2524 In javascript and python there is a memoize (called lru_cache) method that caches calls for functions, the memoization of a no argument function is called the singleton. Static classes run code at the start of your program, while singletons are initialized the first time they are used.
"if only...if only react was dying"...lol you get a million internet points
"btw, I work at Netflix"...
"Oh, you work at Netflix? Why don't you ?"
lol
Decorators are GREAT!!!
"Can we pause for a second? I hate decorators"
Hahahahaha
Python canonically uses exceptions for control flow lmao. Exception zealots in shambles
19:00 I expected facade being something like "struct NodeId(usize);"
Internally it's just an integer, but you can't use it like an integer.
he looks so guilty when talking about factories :P
I've learned a ton from this kinda stuff, so thanks again man. Don't think I can put my 110 wpm to use the same way y'all can.
till now, i don't no crap about programming styles and patterns, I'll code whatever works
On the last point of the facade pattern, NeetCode says "you don't want to think about memory allocation" and here I'm thinking "it isn't difficult at all, there is this thing called realloc." The only reason why it sucks in java is because java does precisely this, remove any need for you to think about memory allocation. Realloc is a better abstraction than java.
Netflix is implementing java today
16:30, Unfortunately I'm the opposite, I suffer from success, but it is what it is.
self head had me out of order for 1 hot minute !
Command and Observer are my personal favs