Based and procedural-pilled take. OOP and FP are basically both attempts to make PP scale more nicely. Let us never forget that in the end, the core of programming is still PP.
“You wouldn’t want `address` to be public, since any other class would be able to access and change it.” *immediately adds getAddress and setAddress methods as “best practice”*
Honestly, that is why i do not like devs that shill OOP as the best thing in the world. They just know how to copy and paste stuff they have seen in books, like "Clean Code™", and love to overcomplicate everything because it is a "Good design pattern". Getters and setters are perfect examples, they are useless in 90% of ocasions, but OOP shillers use them everywhere, even though it does the same thing as just acessing the variable directly.
@@rj7250a it's not even good OOP in my view, it just adds noise. If you evaluate your objects by asking "what should each field/method be able to do" and not just add things because it's what you've been told, you wind up with OOP code that actually takes advantage of encapsulation instead of doing backflips to get around it. So for example, if you have a field with a get/set that just returns/sets the underlying property and NOTHING else... why wouldn't that just be a public field on the object? It's adding lines to just add lines, lol.
Setter- and Getter function are kind of a mess if you ask me. If changing or reading a Class property has to lead to a chain reaction, then they are mostly unavoidable. But, if the property is independed then having to define 2 extra methods just for the user to access them, bloats the Classes code. Its kind of a trade: For a Class user, knowing that changes to an object can only be done via methods is easier to understand. (hence why everyon thinks it is "Cleaner") But for the Class writer, it mostly boats the code and makes things harder to read/navigate. That is, why i love Javascripts 'get' and 'set' function attributes and PHPs __GET and __SET Magic Functions. They let you do all the stuff, that Setter and Getter functions can do, but still present them both combined just as one attribute to the Class User.
There are some usecases : - the user should only get or set the value - you need to do data validation before setting the value In this example it makes no sense
@@zacharychristy8928 I have seen good OOP, but it has been done by pragmatic programers, not OOP shillers. Pragmatic programmers use whatever is the best for the current problem, they do not follow a strict set of principles in a book, as if it was a religious dogma.
Videos about OOP that start with a description of classes, methods, and inheritance are the CS equivalent of starting an essay with "Merriam Webster's defines [essay topic] as"
@@Cookiekeks Everyone does that. At this point, it is either just filler if you know what OOP and PF is, or a video for beginners, which you can find an abundance here.
It always brings me pain, because they (almost) always give class examples of real-world objects. Code is not real life, classificiation is never that simple and clear in practice. I'm not writing code to deal with the different types of apples or fish. I'm handling files, or graphics, or compression, or strings.
@@colemanroberts1102 the emacs lisp book chapter one describes how to read the book; chapter 2 (the first real chapter) describes the data types in fact it says nothing about how the program is evaluated before chapter 10, and the term `s-expression' does not appear (outside of footnotes) before chapter 35 out of 42
i disagree with you: in this case the program is receiving power as input and releasing heat as output, also it's affecting a global variable called "Universe_Entropy". in that case, the perfect functional program is actually:
For me the question boils down to a few simple concepts, 1. The object-oriented paradigm revolves around passing data between unified domains, not necessarily "classes" or "structs". That means you send information back and forth and try to make each specific domain responsible for interacting with that dataset within that scope of states. 2. The functional paradigm revolves around making code immediately obvious in a declarative way, even if that involves using data encapsulation methods and the like. The code must read fluently and it must be immediately obvious the result of an expression not by an individual stream of parts, but by the complete expression (and, of course, it must be decomposable). 3. The procedural paradigm revolves around ordering things in a specific, controlled way, not through statements or units, but through small, specific, determined logical steps that must modify or change the data as they are performed. The scope of a procedural code is always linear and it must be possible to read and understand linearly. To that extent, I can understand that all paradigms employ common concepts and can be summarized to common notions present individually in each of them, but which are not immediately reducible to these, as a complex system. Each of them has its place and I can understand why multi-paradigm languages won and purely functional or purely object-oriented languages became less and less popular.
You cant say that. A mature response is not for the internet! Here you need to pour gas on the fire and say "THIS PARADIGM IS TRASH, THE ONE I LIKE IS BETTER!!!!". Dont be sensible. It's the internet.
can you please elaborate on declarative and imperative code ? is my thinking correct that all declarative code is always abstracting some underlying imperative logic? To me the functional paradigm seems declarative in a way that it abstracts imperative logic. for e.g even when writing an 'imperative' c = a + b, we are abstracting the logic of actual addition on bits away from us in the "+" operator, in turn making it declarative. Also if you've used react can you comment what kind of paradigm its following? is it functional UI or declarative JSX? or something else? I think its time for me to learn haskell lol
I remember once... in haskell I was trying to manage "connection codes" on a server... kinda "low level" in the sense I only used the standard libraries to do it... no frameworks... You start client A, which asks to connect... The server accepts and sends a "code"... You could(out of the loop) send to your friend... Your friend starts client B which send that same code to the server... and then you( and your friend) start a session(in my case... a battleship game) managed by the server in particular... to manage the codes in a server... I essentially needed 2 functionalities: 1. Create a brand new code associated with connection with client A(which just connected asking to make a new connection)... give out the code(so the server transmits to the client A so you can send to your friend) 2. Given a code... return the connection you and your friend can play; I'm not using the haskell notation like I did there. I will do the equivalent in python; def codeManager(): id = 0 connections_to_be_paired = dict() def assignID(connection : TCPConnection): nonlocal id current_id = id id += 1 connections_to_be_paired[current_id] = connection return current_id def pairConnection(pairing_id): try: connection = connections_to_be_paired[pairing_id] del connections_to_be_paired[pairing_id] return value except: return None# This function originally returned Option.... so it was not as sketchy as it is in python return (assignID,pairConnection) That's like.... very old school OO done via closures. People would use a class for this, with attributes "id" and "connections_to_be_paired" private(here they are encapsulated because there's no way to reach them from outside), and public methods "assignID" and "pairConnection" (in my bodge they are the return values). It's not bad... but you had to remember the order of the "methods" that were returned: assignID, pairConnection = codeManager() # assignID, pairConnection... just like the function returns Nowadays I would have wrapped them in a struct equivalent in haskell. The closest thing to class you can get without significantly more workarounds
9:42 fun fact, this code on the screen doesn't actually have any side effects. Python only assumes you use global variables when accessing, when assigning values it creates a local variable, unless you use global keyword
I can handle 2 or 4. I tolerate 8. But I've seen 1 space indenting and I wanted to curl into the fetal position and cry. With 3, you are just pissing off anyone with OCD, lol.
As someone who grew up with procedural programming before OOP truly entered the mainstream, it was interesting and slightly humorous watching OOP grow, suddenly OOP was considered the cool kid, your programming language wasn’t worth considering if it wasn’t object orientated, people couldn’t gush enough over it, it was new and trendy, everything HAD to be OOP, and the young coders were told OOP was like some god that had to be worshipped… now people are realising that while OOP has its uses, it’s not the all singing all dancing paradigm they were told, and can be a steaming pile of hot garbage.
I made a full GUI, multi-option/filter database searcher as my first program. Super snappy, fun to use, it would search "name 04" if you typed "04" etc. It even searched as you type! I asked how I could improve, and they said my use of "goto" was frowned upon because it makes the code hard to understand. It was 1 goto in 46 lines of code with zero functions or classes. To replace it would require breaking the whole thing up into multiple functions which would then "goto" each other. That's my issue with this debate. Functions are just "readable" gotos, and classes are just "readable" functions, and inheritance is just "readable" classes, and programming languages themselves are just "readable" versions of other programming languages. Seems like we're forgetting how to read!
I remember encountering Basic when using a Commodore64 emulator, and found the goto keyword to be really cool and simple. I still kinda miss using it. Goto is just really simple and easy to understand! if condition, go here in this part of the code! if not, just continue on then perhaps when that not statement is done, jump to the end.
I invented a new paradigm called HBO - hairball oriented programming. It consists of one flat function that does everything. Who needs global variables?! ♥
I mean this is how I program... But I avoid repeating the same code, so it becomes a separate function. And if something is clearly delineated from the rest then I also move it into a separate function. The resulting code is more performant, readable, and maintainable than what you get with OOP.
@@etodemerzel2627 yes, the most common pattern unfortunately is called the big ball of mud. I used to write like that back in the day before getting into clean code, architecture and design patterns.
@@farqueueman Well, I'm moving away from Clean Code:tm:, etc. Over the years, I've seen plenty of codebases written with these approaches. All of these codebases are awful: hard to read and navigate, hard to change anything, and, on top of that, they're slow. My "ball of mud" is better in every way.
14:23 yep, that was me. Until I started learning a little bit of Haskell I went "wait if this is rules of functional then what the heck is C" and that's how I learned about the "procedural paradigm"
Here's my paradigm step by step: 1. Solve the problem with messy procedural prototype code 2. Find where the state can be localized and calculated as part of an idempotent function 3. Put const/final on everything I just added if the language is not const by default so I don't trip over myself changing something I otherwise shouldn't 4. See which state is forced to persist after the important function calls and turn those bits of state into a "fake continuation stack" aka internal localized state in an object
Be careful though, as Prime noted in one of his previous video, this approach of incremental fixing toward some local optimum may lead you to find out that it needs a complete rewrite to get the best global optimum solution. However, in my opinion, following paradigms like FP or OOP, or maybe even some custom ruleset, can allow you to get to those local optima quicker, as you train yourself to write paradigm-following code more and more, thus saving you time on incremental fixing in the long run. I think this is also literally the point of Rust, which always seems hard and slow to write, but in the long run you would not only write more correct code first try, but also know writing what could get you more correct.
@@CYXXYC The more sure the requirements are, the closer it gets to that last step. Unfortunately, an ever evolving product with even more unsure product folks keep me from going past step 3 most of the time anyway.
@@SimGunther Hi can you please suggest me some program/system I could write to test different paradigms and even custom rulesets for their tradeoffs and to also upskill myself in different modes of programming/designing systems. TIA
That’s pretty much just called “programming” and how it should always be done. 1) do exactly what you need to do with data. 2) abstract out the common state for compressed and efficient code.
The 'look in less places during side-effect-free FP' is sorta true, in the sense that you don't need to look around for more and more things as you read through a function, the scope only gets tighter and tighter. Like if you write a django view, at any point the request could hit the DB and influence the state of the app, but with Haskell or whatever, unless you're doing side-effects (IO), you can be sure that you only have access to the variables you did at the start of the request. You don't need to check what state the database is in, because it wasn't passed in as a variable. Hard to put into words but it does feel simpler (in certain aspects) compared to the imperative style, even if there's tons of overlaps. Its all about picking your poison for sure.
A Django app was my first large production OOP based thing to work on. It's awful. As you go into things to figure out what's going on it just goes everywhere.
"the scope only gets tighter and tighter." This is kinda the essence of composition in OOP. However it's not that simple, it's usually almost impossible to design code in purely hierarchical manner. Now OOP has produced plenty of great literature how to design your application that can achieve this property - onion/hexagonal/etc architectures are all born from applying the principle of composition to large apps.
@@sk-sm9sh Pure functional is all about composition because there's no way not to do it! Tons of purely hieracrchical projects exist using Elm or Haskell since they don't allow using something you don't pass in (with some tiny exceptions!). You should try out some FP languages if you're curious!
@@TankorSmash here we go again "you should try some FP". From my experience really more people tried it that FP zealots like to think. Just because I tried it doesn't mean I love it already. To me it falls short in many regards.
I hate this generation of devs. Everyone and their mom want to be come „dev influencers“ when they are obviously either just BAD at it, just graduated or are just repeating the bad Medium posts of others. Saying „public static final“ is a OOP pattern is so stupid and then continuing „now I want to show the beauty of FP, but I never used it so sorry I’m advance for my bullshit“ is so ridiculous. „Engineers“, we are laughing stocks
And using print, a function that only has a side effect, as your example for functional programming is just ridiculous. Not to mention that he just writes a wrapper function to print which requires that the argument is string, but passes it directly to the print function which uses dynamic dispatch... The video seems like such a confused mess
I think one of the most overlooked upside of pure functions and immutable variables is the ability to distribute (and cache) their workload. Since you don't have to track a shared state and the output is deterministic all the time it does not matter which machine (or processor, or core) works on your function, the result stays the same.
This is basically the raison d'être of Erlang (and by extension Elixir), it's astonishing how seamless & lightweight process creation is. I'm not aware of any other language where you can be running thousands of threads without even batting an eye.
oop has a problem. creates redundant elements in memory. In many programming languages an object inherits from a base object. Therefore, every time it is instantiated, its elements, properties, methods, special methods, etc. are also loaded. fp doesn't do that. The data is separated from the methods, whether they are pure functions (calculation) or impure functions of actions. In addition, the Earlang virtual machine does one thing, and that is the virtualization of processes, so these are not loaded into the operating system with a life cycle and everything else, they are done within a virtual manager of processes and their Manager and process hierarchical trees
Functional Programming can't work with side effects?, in Haskell (Advanced Pure Punctional Programming Language) you can work with effects using Functor, Monad and Applicative
The vast majority of FP in the wild goes on in imperative languages. Functional-style code in Python and JS is still just a bunch of procedure calls. That's because these are the mainstream languages. (except Excel ofc) But the converse is also achievable. You can take a purely functional language and build stateful, imperative abstractions within it. Btw, you should really try expanding your view. Try learning some variant of Lisp (Scheme seems solid and small-enough) and something with its roots in ML (maybe Elm, since it focuses on FP novices and is very small too)
Common Lisp is best lisp. Scheme is so small that tons of important functionality is implementation dependent. Common lisp has a big standard, but excellent facilities. Also, unhygienic macros, unwind-protect, CLOS, excellent exception handling, optional dynamic scoping, etc. Clojure has neat features and excellent java interop, and scheme is super small and really lets the implementor play around the design space. But common lisp's interactive development is top tier, even among lisp's.
Also if your compiler/interpreter cannot collapse several chained map functions into a single loop it is not really functional programming. It is just functional style. Take SQL for example, you can nest SELECT clauses all you want, but it isn't going to create a bunch of temporary tables if it can evaluate the expression with one index traversal. SQL in that regard is much more functional than the functional part of JavaScript. Of course in SQL you can't store functions as values in tables, so that is where that ends.
@@colemanroberts1102 Common Lisp definitely is more suitable for making relatively fast real-world applications, but for someone more interested in learning programming language paradigms & fundamentals, Scheme's minimalism & simplicity is actually a benefit, e.g. I couldn't wrap my head around the concept of an "object" until I had to implement them from scratch in Scheme as a CS freshman. Other things we implemented in it are a type system (using cons cells), lazy evaluation (using lambdas), DSLs, a filesystem, a relational database management system, a meta-circular evaluator, and just about every data structure known to mankind. It's the ideal teaching language imo. Also, Racket is a great scheme variant that offers the best of both, with a comprehensive standard library, package manager, build tool, and top-notch documentation, it even comes with its own IDE, the only thing it really lacks is speed.
@@ultru3525 minus the meta-circular evaluator, all those things are just as easily done using the same techniques in Common Lisp. You are correct about scheme being a better platform for learning to implement a language though.
@@colemanroberts1102 Oh yeah, I wasn't trying to imply you can't do those in Common Lisp, my point was that all of those can be done in Scheme, in spite of its limited facilities. And as a teaching language, I feel like Common Lisp suffers from the same problem as Java: too many options that are useful in production, but can only serve as a distraction when you're trying to focus on the fundamentals. Common Lisp was the second language I got comfortable with after Python, and while spending two days trying out all permutations of `loop` keywords was interesting, it wasn't really the best use of my time. With Scheme, it's practically impossible to "waste" time learning language-specific features, you'd have to delve deep into the intricacies of continuations for that.
I bet this Codepersist guy will look back at this video in a few years with embarrassment about all the misconceptions he exposed. But I guess putting together the video taught him something, so it's all part of the journey. But my concern is, how many other junior devs he'll confuse through his public learning journey.
Encapsulation is not about data at all. Encapsulation is about implementation hiding -- David Parnas called it information hiding, really meaning hiding the information about the detail of implementation. Thinking private is about encapsulation is wrong. But that is not the only reason private is wrong. Hiding fields from subclasses is wrong.
The great thing about Python is that if I manually set the recursion limit, I can choose which level of abstraction I live on. I can get most of FORTH's extensibility if I want it, or I can have completely pre-cooked loops if I want them as well.
5:20 I find it more senseful for Rust to use Wrappers(sounds pretty similar to encapsulation for me) not to protect sensitive data, but to safeguard the data's/ or type validity. For example, a non-empty string which is also a valid email. This Data-Type we will call it "Email". now, everytime you use Email as a func's argument, you can be a hundred percent confident that the function is both receiving a valid and non empty e-mail. And furthermore, when copy-pasting the function (let's say the version that use a String as argument instead of Email), you won't accidentally forget to parse the string, or even do a double/multiple parse(one inside the function, and another at the callsite(s) because you forgot the function would parse the String automatically for you)
Functional programming maps so well to many things in mathematics and Computer Science but OOP maps well to the real world. For me though, FP is good when I'm building and all the ideas are in my head but if I come back to it three days later I'm confused.
@@etodemerzel2627 ehhhh...? I know you can create custom types and there is the data keyword in Haskell but... I don't use it... maybe that's the world I'm from.
It’s a grave error to try and map the “real world” to code. Code exists entirely in 0’s and 1’s. There is no “chair” that inherits from “furniture”. There is only data.
oop has a problem. creates redundant elements in memory. In many programming languages an object inherits from a base object. Therefore, every time it is instantiated, its elements, properties, methods, special methods, etc. are also loaded. fp doesn't do that. The data is separated from the methods, whether they are pure functions (calculation) or impure functions of actions. In addition, the Earlang virtual machine does one thing, and that is the virtualization of processes, so these are not loaded into the operating system with a life cycle and everything else, they are done within a virtual manager of processes and their Manager and process hierarchical trees
Good video. Real quick, total noob here, why would you avoid inheritance? It just strings too many things together causing bug enhancing dependencies? It seems like such a cool idea to be able to create your world through a process similar to nature. I've had waking dreams of using a few primal classes and generating a universe via inheritance. I guess it isn't as pretty irl?
This is from a Python point of view. The only reason to use Classes is to simplify a complex object into JSON basically a dictonary. Sometimes it's easier to retrun a dictionary with a Class than a function. Functional programming only to not change the initial state of that object (Immutablity). Procedural programming for everything else. The thing I like about Python is that you can abstract basic things with a context manager. This is like procedural programming without having to worry about the end. For example, writing to a db. I don't have to procedurally say...insert, commit, close, roll back if something went wrong. The context manager handles all of that. Also, If you need to go for speed (concurrency or multi-processing or some combo of the two) it's simple enough to work with if you use a Class to make your iterable and an asyncronous function to blast the hell out of it all while avoiding bugs caused by inheritence/state/polymorphism because you're at least adhering the the principle of immutability and can thus better deal with the expected output or raise an Error/Warning.
Disagree here. FP is an approach. A mindset. FP promotes high cohesion, low coupling, side-effect management, state management, etc. all ideas the React borrows. Unix piping data around is FP. The web's statelessness is FP. XML/HTML is just LISP with syntax sugar. It is FP. Just because you create a class doesn't mean you're not following FP principles. And just because you use Monads like Promise doesn't mean you are.
This needs to be a short... "What have you dobe here? What have you done? This things like 12 characters here"... And then another short that overlaps a little and includes the Fibonacci bit
Would somebody mind explaining what prime meant by properties being one of the big problems with OOP/classes? Im developing a language and I'm currently writing a EBNF/grammar for the language. I think some approaches I have come up with can avoid OOP confusion in some areas but I'm stuck on what prime meant by the properties comment, could be interesting?
I can't work out if the diagram at 4:30 is an inheritance diagram. If so that is an insane hierarchy. Too often people use really bad examples in OO and say OO is at fault. Subclassing is a very tight relationship and should be used carefully and sparingly. Car --> Toyota --> Truck is not an inheritance hierarchy, what is it? It does not make any sense really on any other level.
People think of inheritance as a mechanism for building up taxonomies, which it is absolutely not. They also get the whole reuse-through-inheritance completely wrong. They think "Oh, I want to reuse that class, so I'm gonna inherit from it." But the point of inheritance is not to inherit functionality, but to implement towards an interface, so that the code that _operates on that interface_ can be reused. Also: Square aren't rectangles as long as you're not willing to accept squares with differing side lengths.
@@pillmuncher67 I think you are confusing taxonomy and reuse. Inheritance is very much about type taxonomies. I agree that thinking "I want to use/reuse that class, so I must use inheritance" is completely wrong. However, if we use inheritance for taxonomies, correct reuse will follow. That is use interface inheritance, not implementation inheritance. But Java is completely wrong in separating interfaces from classes. All classes implicitly have an interface. There is no need for an explicit interface concept. Inheritance constrains the parent class. Thus all squares are absolutely rectangles. The definition of square conforms to rectangle -- a shape with four sides the opposites are parallel and all angles are 90º. But square also conforms to rhombus, a quadrilateral that has all four sides equal. A square can thus be defined by multiple inheritance between rhombus and rectangle which defines square. Remember inheritance deals the the abstract definition of entities. Now there can be implementation attached, but not fully specified, and this is what is wrong with Java interfaces. See chapter 24 of: bertrandmeyer.com/OOSC2/ which discusses correct use of inheritance.
@@ijoyner Yes I know that book, I have it on my shelf. There are points where I agree with Meyer and others where I don't. But ask yourself: If you have a Rectangle class with a method set_width(n:int) and the post condition is that only the width is changed but not the height - what could possibly be the meaning of this for a square? Can squares have sides of different length? Should an error be raised? Should you write an if statement to test if a rectangle is, in fact, a square? Why not just make a new Square class that inherits from, say, Shape, and that holds a private reference to a Rectangle object to which it then delegates? Like, if the side length has to be changed we call both set_width() and set_height() with the same argument. One must always look at the problem a specific technique is trying to solve. In the case of OO, as it occurs in the wild, the problem is to have polymorphic method dispatch over one parameter at runtime, so we can add new things without having to go through the code and add case blocks to switch statements. Everything else (entities, taxonomy, ...) is marketing.
@@pillmuncher67 The meaning for square is that both set_width and set_height set the length of the side of the square since the invariant of square is width = height. Perhaps we inherited from a general quadrilateral which has sides l, r, t, b. In that case for square l = r = t = b. Perhaps we should not label the sides left, right, etc since we might rotate the shape. "Should you write an if statement to test if a rectangle is, in fact, a square?" No need. That is expressed in the class invariant. That is essential properties of objects are separated from the code. That is the purpose of redefine, select, and undefine in Chapter 15. "Everything else (entities, taxonomy, ...) is marketing." Not at all. Taxonomy is a scientific approach to understanding things, the similarities, the differences, the constraints that distinguish a subclass from parent classes and other subclasses. Taxonomy is almost the most important part of the technique. But taxonomy is not exclusive to OO and can be used in other paradigms like FP. Type taxonomy is important. But it should not be used for reuse first. Let the reuse occur as a pleasant consequence. That is where people go wrong. These are all considerations and considerations that inheritance brings up. That does not mean it is wrong -- you will face these questions whatever. You could just implement as completely independent entities, but then the relationships between the shapes, the commonalities and differences won't be apparent and no reuse will result. (And don't use '()' to signal 'function' -- '()' is a primitive call operator and ugly Hungarian notation exposing implementation.)
@@pillmuncher67 I think the other aspect where people go wrong about inheritance and taxonomy is they are thinking that B inherits from A means B has added to the functionality of A. But A expresses many possibilities and B is actually taking away from that. When people think of implementation inheritance they think that is everything A does, and we add what B does. Well, no, we put the general facilities in A that cover a lot, defaults. The job of B is to subtract from that. Again to express by constraints why B is different to other subclasses of A. That is interface inheritance (but again Java is wrong in separating the notion of interface from class, and that in itself results in very artificial inheritance hierarchies).
Not true about procedural. There is a thin mapping layer you can actually get to. So if you have a clear algebraic structure set up (association and id’s), you can pretty much get away with behavioral and dataflow modeling. But I get why one might perceive this as a procedural approach. The functions are pure, though.
I like DOP (Data oriented programming). Understand your data and the necessary transformations you need for that data. Then your code will solve your problem (and nothing else). Writing it will be straightforward.
I am learning DOP now, but is only popular between game devs now, since it uses very unconventional syntax and is harder to extend. I have seen languages like Zig, that are trying to make DOP more ergonomic, you can write a conventional struct, then use a function from the stdlib, so it will be converted to a data oriented pattern under the hood.
@7:50 i whole heartedly agree, but i think we all do this. its just more, the leaf nodes, of what you use, should contain all of ur behavior and modifications... im trying to think about how i love protocols, and typing but hate inheritance... having a protocol is such an easy way to perform a type union, even if u have to define an interface. it prevents messy if logic and gives a path forward, with only 4 more lines of code. unfortunately, today, i found myself liking asserts as a way to force typing, when underneath an optional, when unfortunately i needed an optional, as i have a conclude method which creates the return object, but theres a step which only executes for some of the paths. turning on mypy for the typechecker shozwed me how much happy path cofing i was doing 😮😅
I think the intention of Functional Programming is for solving problems in a mathematical way. Like you said mainstream functional programming is just organizing your code in procedures without side effects. Same with OOP where Alan Kay was like it's more about the messages than the objects themselves, but objects are just another way of organizing your procedures. Procedures FTW!
"Address and SSN are sensitive data therefore they're private, but name isn't so it can be public" That's…not the goal… It's a terrible misunderstanding of the point of encapsulation!
Correct me if I'm wrong but to me encapsulation is used to better monitor the use/change of variable and avoid messy classes using eachother variables directly without a going through a function you can use to understand the situation better if, for example, you're debugging code
@@Malenbolai Basically the goal is that everything outside of a class should not be concerned about its inner workings. That ensures that the data within is always treated consistently and according to the desired logic. So yeah, it's pretty close to what you said. It's not about what the end user can do, it's about what the _developer_ can do.
ff vs oop vs pp really depends on what you're making and how it should read when you're debugging it. IMO it should be natural. If you're hesitating on paradigms, patterns, syntax, etc. you need way more practice. If you need hierarchy use OOP, if there are clear steps, make it obvious, and I guess otherwise try to preserve immutability at all costs.
2:37 Crazy default inheritance system in Python is precisely what made me avoid OOP in Python and almost made me quit Python as a programming language. *BUT* Multiple conditional inheritance in Lua, explicitly implemented in the program by using the Lua language itself and exactly according to my special needs, together with memoization, once made me write an elegant, fast and memory-friendly pool of procedurally-generated sprites for real-time visual effects for a video game. Because I designed and implemented the inheritance mechanisms myself, I used it as intended and wasn't struggling with some bad generic compromise made by language designers. *I made up my own OOP mechanism according to the type of problem I had to solve, and not the other way around.* Building your own inheritance system is very easy in some langages, requiring no special keywords nor classes, simply by using the available structures in order to define closures and conditional calls, and let your decide exactly what is inherited, under what conditions, and how exactly (priorities, copies, references, memoization…), even at runtime. But, yeah, what I have learned from this is: *use FPL to define OOP-like multiple - or even concurrent - inheritance as you like* , but *do not try to practice FP with a rigid OOPL* unless you plan to hate yourself.
seeing this video 7 months later and knowing that he hasn't moved to wayland makes me think problably 2024 is wayland's year, as 2023, and as 2022, and as 2021...
I'm not convinced that the person in the video (not Primeagen) actually knows what he's talking about. Properly formatting the example code would improve the appearance of reliability.
I'm a big fan of FP, but I do agree with the general idea that multiparadigm is probably the most optimal way. Engineers in all disciplines are taught to "pick the right tool for the job", but software engineers don't seem to take this to heart and they end up being hellbent on proving one approach right. You're taught to break down your complex problem into a bunch of simpler ones, and picking a paradigm is the same deal: some of these sub-problems will fit one paradigm better than the others. Procedural rules tho'
I heard an interesting take on this whole paradigm war: - Use object-oriented ideas (encapsulation, representation hiding) for your core data - Use functional programming (immutability and pure functions) for your business logic - Use procedural programming for the tasks that talk to external systems I might have forgotten a paradigm or two, but I got the central message right: use the central ideas of each paradigm in those parts of your code where those ideas accomplish some real, useful benefit. Also, OO people talk about data hiding. That is wrong: getDayOfMonth will return data. It cannot be hidden and it should not be hidden; it is the essential point of a Date object. What _can_ and probably _should_ be hidden is how this value is arrived at, and how this small integer is represented internally. Likewise, the exact representation of nodes in a red-black (or AVL, or B) tree is probably not something the user should know about; they should just have a Map interface and a logN performance guarantee provided the comparison function is sane. But the point is not to hide data, the point is to hide how data is represented. TL;DR: it's _representation_ hiding, not data hiding.
> Also, OO people talk about data hiding What people talk by data hiding is exactly what you mention later - hiding internal representation, thus establishing encapsulation. Nobody has ever said that you should not have methods that return values.
I strongly object to distinguishing OOP/FP on auto-completion. Packages export constants, functions, types, and even function-types (if you export mutable values, you gotta go see Karen, my man), where class instances export mutable values, and bound-functions (methods). It’s the same thing, with different flavors. It’s just namespaces, at the end of the day. What people who understand the issues are talking about when debating OOP/FP is whether the ideals of either are worth the tropes. You can write shallow OOP, preferring composition over inheritance. You can also write OOP with deep caverns of multiple inheritance, bottomless pits of overrides and monkey patching, and always the subaural drumbeats of doom beating out “Meta..classes…meta…classes…meta…classes…”. FP is more dependent on the compiler to do well, but Rust’s iterators are a FANTASTIC example of a functional paradigm that has great support from the compiler, is fast, and is still very clear and modular. Honestly the biggest thing for me: write the code you need to get the job done, keep it clean and responsible, pick up your errors like a decent human being, and check for null/nil before dereferencing. It won’t matter if you don’t have setters and getters on every public property or if you mutate data in place or return an altered copy. What matters is that future you and friends will be able to observe what is changing and where and why. So yeah. Hardcore object to insisting a pattern is too hard or unreasonable without understanding it or where the payoff is. That said, copying every value in your array to return a modified copy - or maybe even not - can obviously be an expensive price to pay for making your code more readable.
Overall, I do agree, but maybe I’m just…not a developer? (I worked as a dev in the past, though) You see, for me, classes of equivalence matter a lot. So to keep track of that, you must follow stricter rules. Pure functions definitely help with that! Too much perfectionism..sorry. I don’t like (regular) dev work. Waay to much mambo-jumbo and spaghetti.. I’m more conceptual..😢 I suffer..
@simoninkin9090 , I think I'm saying that, but with more mumbo-jumbo. I'm pointing out that there are some real considerations, but if all you care about is being able to put a name followed by a dot, and have your IDE suggest completions, you can do that in either paradigm, not just by using classes.
@@samhughes1747 I totally agree about the necessity of good namespacing. The question is ultimately about composition. Sure, objects can also compose (I did some of that a few times), but then these objects must comply with the similar rules as pure functions. And it is actually harder to work with.. you might want to call these “function objects”. There is an override in cpp for such cases. You can definitely do everything in either paradigm. In fact, as a _professional_ (the one I do not ever want to become…because screw the payoff! =)) ) you should be able to slide though the codebase, written by other people and follow the paradigms in the code itself. All significant modifications must be accepted by the tech lead, etc.. What I’m talking about is kinda R&D…or whatever. Maybe it’s not that practical in real life (though, the finished code can be optimized and quite fast..just that nobody would understand it..that’s all..), but I honestly don’t care. There is no freedom in commercial programming. For some people it’s..well - just a job. (Don’t get me wrong though, I’m quite average). Yet, for me, it’s more like a tool for understanding deeper realms of our universe. Not freakin’ kidding! =))) FP maps very well onto many concepts in modern physics (I’m not a physicist, but I’m really curious and can’t stop digging for answers..or rather - more questions) P.s. never actually touched Rust myself, but the iterator pattern sounds like it’s the same as STL iterators. In fact, the essence of C++, the Standard Template Library is also functional! But it’s a mess! If you compare that even to F# (hate dotnet though..or any vm, which is not llvm. Bytecode really sucks!) or even better - Elixir (possibly the cleanest syntax I’ve seen in a programming language)
@@simoninkin9090, gotcha. I was mostly complaining about the idea of dismissing FP based on auto-completion advantages, and emphasizing that OOP and FP both have namespaces that lend to that auto-completion. It’s one of the most common complaints about FP I hear, and it’s nonsense. Meanwhile, yeah, I work on a team with a variety of skill and discipline levels. We have a lot of code that drags a logger of some kind through the whole program, and same for whatever database or API client is being used. It’s nauseating, but I’d the program does what it’s supposed to, and if there’s tests to PROVE that pile of tech-debt does what it’s supposed to, I’ll sign off. I care far more about my coworkers doing good failure-mode analysis and proper error-handling. My team lead will back me up on complaints about squashed errors or untyped interfaces, but he isn’t going to back me up on holding work hostage to a paradigm that is generally widely misunderstood. I absolutely am convinced by the ideals of the FP paradigm. In practice, like you say, there’s everyone you work with, and you’re writing code for them to read too.
I think classes are ok way to abstract the different designs but the issue is when people teach these abstractions where you have to drive it from some real life concept and that doesn't work with software because you have to understand the context where your application is running. You abstract the application to memory etc and not some stupid animal that becomes ketchup.
7:50 What I hate most about inheritance is being able to override things multiple times. Something should either be abstract or maybe virtual with a default implementation, and you only override it once. Else something like this could happen. There's a class A. Class B inherits class A, since it's basically the same, but almost everything is slightly different. Class C inherits from class B, since it's basically the same, but almost everything is slightly different. Why not just let class C already inherit from class A and duplicate the one method from class B, which is still the same? Or even better, turn this method into a function. Maybe just create an abstract super class for A, B and C, all three of them, and only make the parts, which are not exactly the same abstract methods?
Just write lisp and lose all ability to distinguish between the two consepts. Because the functions don’t share the same namespace with values - symbol can have both. And functions are valid values as well!
12:50 Yeah I like C# extension methods for this reason. print(derpNamespace.derpFormat("Surprise MF!")); becomes print("Surprise MF!".derpFormat()); and you don't need to remember which file/namespace derpFormat is in
I’ve actually stopped using encapsulation, at least in the explicit, language supported sense. Now I essentially make everything public and static, and I simply use organization and architecture to implicitly enforce simple rules about what code is allowed to call specific methods or access specific data. I like to think of it as Extreme Domain Development.
Implicit rules are ok if your team is small. How you're going to enforce it if there is 100+ people working on project? Java has so many access control keywords for large teams and indeed they are unnecessary for small teams but for large organizations being able to specify access-control is very helpful as it's makes enforcing ownership easier. For instance I tell that I own this class, and I put everything as private. Now it is easily guaranteed that no one will access the private vars without modifying the file. Since I observe changes to my file I can easily notice if someone tries to get stuff that I marked as private.
@@sk-sm9sh Having 100+ people working on a single codebase is becoming pretty rare these days. I only work on two types of projects right now - personal projects with one developer, or micro services with 1-3 developers. But, hypothetically, if you have more developers, you don’t want them checking in code that hasn’t been reviewed by at least one senior developer. At that point, it would be their job to ensure that the code is in the right file, and that it either has zero dependencies, or that it only references code that is located in its sub-domain. Think of it as an org chart. If everything is organized properly, you don’t normally have salespeople talking directly to developers. Technically, you can, and it’s not really the end of the world if it happens. But ideally, you want that information to go through some sort of management structure, so that resources can be prioritized properly. I essentially do the same thing with software architecture. I have components that actually do work (read files, send messages, query data, etc.) but make no decisions, and I have “management” components that wait for event notifications from sub-components, and then send commands to other sub-components. I also try to keep the number of sub-components managed by a single management component to a minimum, usually no more than 2. This makes design work fairly trivial, and when something doesn’t work properly, it’s pretty obvious where the problem is. Although I’d have to say that this approach really does make things so simple that code often works as expected on the very first try.
It doesn’t really matter if you up the abstraction level, or go low level. At some point you’re gonna converge to a pure categorical realm. So yes, you are correct! Everything is just a category!
I've recently faced the same screen tearing issue on my desktop (nvidia rtx 3070). Arch linux wiki suggested to add a "metamodes" option on "Screen" enabling NVIDIA "ForceFullCompositionPipeline". The option did fix the issue but it also made the refresh rate feels slow (feels because it was still 60Hz but it was looking like it was 30Hz). In order to fix that, I've had to allow non-edid modes by adding the Option "ModeValidation" "AllowNonEdidModes" in the "Device" section, and I've had to set the resolution to an XServer resolution and not use the EDID suggested one. That was quite tricky, in the "Device" section I added an Option "ModeDebug" "true", read the /var/log/Xorg.log, and I've noticed there were 2 mode entries for the resolution 3840x2160: "3840x2160_60" and "3840x2160_60_0", the former was from EDID and the latter from XServer. The final config looked like this: Section "Device" ... Option "ModeValidation" "AllowNonEdidModes" EndSection Section "Screen" ... Option "metamodes" "3840x2160_60_0 +0+0 { ForceFullCompositionPipeline = On }" EndSection As my sight is no longer great, I've also set the DPI to 192 by adding the following line to my .Xresources file: Xft.dpi: 192 I hope that it helps you (or anyone else)
All these languages ages are about handling complexity. Whenever paradigm is simple and accessible enough for people to take advantage of it, that’s what matters.
Most of my JS code these days is based upon async iterators. Stack a bunch of async generators on top of each other, pull an item from the bottom, and the generators yield all the way down. Each generator is self-contained and a mix of procedural and declarative. Debugging is easy.
The point of encapsulation is to avoid your code from becoming spaghetti where everything depends on everything. Getters and setters allow that to happen again.
@@falsemcnuggethope yes because then you don't have to call setters to set the value of values anymore. the return type of getters depends on the type of the field. and if that ever changes, now you have to change both the getter and everywhere you used it. it changes nothing. or perhaps i am misunderstanding your argument. how exactly do getters reduce coupling?
@@SArthur221 having only a getter enforces that the dependency is a one-way pasta string, but it's still a pasta string. If you have too many of wrong ones like that, you end up with spaghetti, just as you do if you access member variables directly.
@@zacharychristy8928 "Premature optimization". I suggest you look up the full phrase and the context... This phrase is basically invalid these days. Please, stop using it.
@@etodemerzel2627 Nope! Still totally valid. Optimizing your solution almost necessarily makes it less general. Doing so before you have a complete picture of the problem and requirements is at best a waste of time, at worst a disastrous foundation to build your program on. Your choice of paradigm should be shaped like the problem itself, and when the problem isn't easily modeled by simply abstracting it as "data and transformations" (which describes plenty of problems) then DOD isn't a great fit.
@@zacharychristy8928 So, I agree with you that the notion of "premature optimization" is still completely valid (actually, it just doesn't make sense to say that it's "invalid", it's not a timing problem). But at the same time I think your statement is complicated, all computing problems can be boiled down to "data and transformations", literally all of them. Data-oriented programming is an excellent way to solve many architectural problems while also solving performance problems (of course, it doesn't solve all or is ideal for all types of problems, but overall it can solve most of them efficiently and cleanly).
@@zacharychristy8928 Aren't all computing problems basically "Data and Transformations"? Also I do not see the problem will less generalized solutions. Indeed most solutions should be completely specialized in my opinion.
Either the video is made by an amateur or it's supposed to be a mockery of another video... made by an amateur. As we say back home "if you'd have shut up, you'd have remained a philosopher".
Why on Earth would anyone use multiple inheritance? It makes debugging a lot harder. Instead of going up a linked list of parent classes when looking for a bug, you have to do a tree traversal over a branching structure of parents. Which would be tolerable if debugging was done by machines. But humans are barely attentive enough to properly follow a linked list while debugging. The chance that a developer will traverse a tree of classes in an efficient manner is more or less zero percent. Is there some use-case I'm not aware of where interfaces would work worse than multiple inheritance?
I've used it once, to splice together a status class from multiple partial status classes in C++. It worked very well, but those were also glorified structs.
This could be extremely elitist but I don't trust someone's videos if they mess up their indenting and code so blatantly. Like I even doubt that they code at this point. There's just no way.
Based and procedural-pilled take. OOP and FP are basically both attempts to make PP scale more nicely. Let us never forget that in the end, the core of programming is still PP.
It always has been.
@@ThePrimeTimeagen PP 🧑🚀 🔫 👨🚀
@@ThePrimeTimeagen All bow to the might PP. 🍆
You knew what you were going to do when you were typing that.
There is nothing wrong with liking PP.
“You wouldn’t want `address` to be public, since any other class would be able to access and change it.”
*immediately adds getAddress and setAddress methods as “best practice”*
Honestly, that is why i do not like devs that shill OOP as the best thing in the world.
They just know how to copy and paste stuff they have seen in books, like "Clean Code™", and love to overcomplicate everything because it is a "Good design pattern".
Getters and setters are perfect examples, they are useless in 90% of ocasions, but OOP shillers use them everywhere, even though it does the same thing as just acessing the variable directly.
@@rj7250a it's not even good OOP in my view, it just adds noise. If you evaluate your objects by asking "what should each field/method be able to do" and not just add things because it's what you've been told, you wind up with OOP code that actually takes advantage of encapsulation instead of doing backflips to get around it.
So for example, if you have a field with a get/set that just returns/sets the underlying property and NOTHING else... why wouldn't that just be a public field on the object? It's adding lines to just add lines, lol.
Setter- and Getter function are kind of a mess if you ask me.
If changing or reading a Class property has to lead to a chain reaction, then they are mostly unavoidable.
But, if the property is independed then having to define 2 extra methods just for the user to access them, bloats the Classes code.
Its kind of a trade:
For a Class user, knowing that changes to an object can only be done via methods is easier to understand. (hence why everyon thinks it is "Cleaner")
But for the Class writer, it mostly boats the code and makes things harder to read/navigate.
That is, why i love Javascripts 'get' and 'set' function attributes and PHPs __GET and __SET Magic Functions.
They let you do all the stuff, that Setter and Getter functions can do, but still present them both combined just as one attribute to the Class User.
There are some usecases :
- the user should only get or set the value
- you need to do data validation before setting the value
In this example it makes no sense
@@zacharychristy8928 I have seen good OOP, but it has been done by pragmatic programers, not OOP shillers.
Pragmatic programmers use whatever is the best for the current problem, they do not follow a strict set of principles in a book, as if it was a religious dogma.
Videos about OOP that start with a description of classes, methods, and inheritance are the CS equivalent of starting an essay with "Merriam Webster's defines [essay topic] as"
Go read any lisp book ever written. I will bet a million dollars chapter one describes how s-expressions work.
What's wrong with describing it like thaf?
@@Cookiekeks Everyone does that. At this point, it is either just filler if you know what OOP and PF is, or a video for beginners, which you can find an abundance here.
It always brings me pain, because they (almost) always give class examples of real-world objects. Code is not real life, classificiation is never that simple and clear in practice. I'm not writing code to deal with the different types of apples or fish. I'm handling files, or graphics, or compression, or strings.
@@colemanroberts1102 the emacs lisp book chapter one describes how to read the book; chapter 2 (the first real chapter) describes the data types
in fact it says nothing about how the program is evaluated before chapter 10, and the term `s-expression' does not appear (outside of footnotes) before chapter 35 out of 42
The perfect functional program accepts no input besides power and produces no output besides heat.
Hold on we accept input... hold on the computer accepts power and moves the computer into another world where input has been received. 😂😂😂😂😂😂
Finally a language to surpass Haskell
i disagree with you: in this case the program is receiving power as input and releasing heat as output, also it's affecting a global variable called "Universe_Entropy".
in that case, the perfect functional program is actually:
For me the question boils down to a few simple concepts,
1. The object-oriented paradigm revolves around passing data between unified domains, not necessarily "classes" or "structs". That means you send information back and forth and try to make each specific domain responsible for interacting with that dataset within that scope of states.
2. The functional paradigm revolves around making code immediately obvious in a declarative way, even if that involves using data encapsulation methods and the like. The code must read fluently and it must be immediately obvious the result of an expression not by an individual stream of parts, but by the complete expression (and, of course, it must be decomposable).
3. The procedural paradigm revolves around ordering things in a specific, controlled way, not through statements or units, but through small, specific, determined logical steps that must modify or change the data as they are performed. The scope of a procedural code is always linear and it must be possible to read and understand linearly.
To that extent, I can understand that all paradigms employ common concepts and can be summarized to common notions present individually in each of them, but which are not immediately reducible to these, as a complex system. Each of them has its place and I can understand why multi-paradigm languages won and purely functional or purely object-oriented languages became less and less popular.
You cant say that. A mature response is not for the internet! Here you need to pour gas on the fire and say "THIS PARADIGM IS TRASH, THE ONE I LIKE IS BETTER!!!!". Dont be sensible. It's the internet.
can you please elaborate on declarative and imperative code ? is my thinking correct that all declarative code is always abstracting some underlying imperative logic? To me the functional paradigm seems declarative in a way that it abstracts imperative logic. for e.g even when writing an 'imperative' c = a + b, we are abstracting the logic of actual addition on bits away from us in the "+" operator, in turn making it declarative.
Also if you've used react can you comment what kind of paradigm its following? is it functional UI or declarative JSX? or something else?
I think its time for me to learn haskell lol
I remember once... in haskell
I was trying to manage "connection codes" on a server... kinda "low level" in the sense I only used the standard libraries to do it... no frameworks...
You start client A, which asks to connect...
The server accepts and sends a "code"...
You could(out of the loop) send to your friend...
Your friend starts client B which send that same code to the server...
and then you( and your friend) start a session(in my case... a battleship game) managed by the server
in particular... to manage the codes in a server... I essentially needed 2 functionalities:
1. Create a brand new code associated with connection with client A(which just connected asking to make a new connection)... give out the code(so the server transmits to the client A so you can send to your friend)
2. Given a code... return the connection you and your friend can play;
I'm not using the haskell notation like I did there. I will do the equivalent in python;
def codeManager():
id = 0
connections_to_be_paired = dict()
def assignID(connection : TCPConnection):
nonlocal id
current_id = id
id += 1
connections_to_be_paired[current_id] = connection
return current_id
def pairConnection(pairing_id):
try:
connection = connections_to_be_paired[pairing_id]
del connections_to_be_paired[pairing_id]
return value
except:
return None# This function originally returned Option.... so it was not as sketchy as it is in python
return (assignID,pairConnection)
That's like.... very old school OO done via closures.
People would use a class for this, with attributes "id" and "connections_to_be_paired" private(here they are encapsulated because there's no way to reach them from outside), and public methods "assignID" and "pairConnection" (in my bodge they are the return values).
It's not bad... but you had to remember the order of the "methods" that were returned:
assignID, pairConnection = codeManager() # assignID, pairConnection... just like the function returns
Nowadays I would have wrapped them in a struct equivalent in haskell. The closest thing to class you can get without significantly more workarounds
Woah woah, don't be writing Python out of nowhere like that! There are children around here!
@@zyriab5797 do you really want it in haskell?
I almost spit out my drink at 'leave room for the Holy Spirit' and 'Fibonacci Indenting' 😂
9:42 fun fact, this code on the screen doesn't actually have any side effects. Python only assumes you use global variables when accessing, when assigning values it creates a local variable, unless you use global keyword
Disgusting
that’s so weird. i didn’t know that
3 Space indenting is going to be my new styling guide. It is the informal acknowledgement that you don't care about the well being of others.
I can handle 2 or 4. I tolerate 8. But I've seen 1 space indenting and I wanted to curl into the fetal position and cry. With 3, you are just pissing off anyone with OCD, lol.
@@pharoah327 Aw man, I've started to really like the spacing with 3 space indenting. Did not know it was hated
As someone who grew up with procedural programming before OOP truly entered the mainstream, it was interesting and slightly humorous watching OOP grow, suddenly OOP was considered the cool kid, your programming language wasn’t worth considering if it wasn’t object orientated, people couldn’t gush enough over it, it was new and trendy, everything HAD to be OOP, and the young coders were told OOP was like some god that had to be worshipped… now people are realising that while OOP has its uses, it’s not the all singing all dancing paradigm they were told, and can be a steaming pile of hot garbage.
Anyone else make a bet with themselves how long it's going to take for the costco alert to go off and remind prime to turn off his alerts?
Nice idea, I'll try next time and give my result ^^
This made me laugh more than it should've.
i love you
I made a full GUI, multi-option/filter database searcher as my first program. Super snappy, fun to use, it would search "name 04" if you typed "04" etc. It even searched as you type! I asked how I could improve, and they said my use of "goto" was frowned upon because it makes the code hard to understand. It was 1 goto in 46 lines of code with zero functions or classes. To replace it would require breaking the whole thing up into multiple functions which would then "goto" each other.
That's my issue with this debate. Functions are just "readable" gotos, and classes are just "readable" functions, and inheritance is just "readable" classes, and programming languages themselves are just "readable" versions of other programming languages. Seems like we're forgetting how to read!
I remember encountering Basic when using a Commodore64 emulator, and found the goto keyword to be really cool and simple. I still kinda miss using it. Goto is just really simple and easy to understand! if condition, go here in this part of the code! if not, just continue on then perhaps when that not statement is done, jump to the end.
I invented a new paradigm called HBO - hairball oriented programming. It consists of one flat function that does everything. Who needs global variables?! ♥
I mean this is how I program... But I avoid repeating the same code, so it becomes a separate function. And if something is clearly delineated from the rest then I also move it into a separate function.
The resulting code is more performant, readable, and maintainable than what you get with OOP.
@@etodemerzel2627 yes, the most common pattern unfortunately is called the big ball of mud. I used to write like that back in the day before getting into clean code, architecture and design patterns.
@@farqueueman Well, I'm moving away from Clean Code:tm:, etc. Over the years, I've seen plenty of codebases written with these approaches. All of these codebases are awful: hard to read and navigate, hard to change anything, and, on top of that, they're slow. My "ball of mud" is better in every way.
@@etodemerzel2627 lol
DEZIGN BATTERN YAS
14:23 yep, that was me. Until I started learning a little bit of Haskell I went "wait if this is rules of functional then what the heck is C" and that's how I learned about the "procedural paradigm"
14:06 You absolutely NAILED IT, I love this so much. That's exactly it. This is why I love your channel, thank you so much
Here's my paradigm step by step:
1. Solve the problem with messy procedural prototype code
2. Find where the state can be localized and calculated as part of an idempotent function
3. Put const/final on everything I just added if the language is not const by default so I don't trip over myself changing something I otherwise shouldn't
4. See which state is forced to persist after the important function calls and turn those bits of state into a "fake continuation stack" aka internal localized state in an object
Be careful though, as Prime noted in one of his previous video, this approach of incremental fixing toward some local optimum may lead you to find out that it needs a complete rewrite to get the best global optimum solution.
However, in my opinion, following paradigms like FP or OOP, or maybe even some custom ruleset, can allow you to get to those local optima quicker, as you train yourself to write paradigm-following code more and more, thus saving you time on incremental fixing in the long run. I think this is also literally the point of Rust, which always seems hard and slow to write, but in the long run you would not only write more correct code first try, but also know writing what could get you more correct.
@@CYXXYC The more sure the requirements are, the closer it gets to that last step. Unfortunately, an ever evolving product with even more unsure product folks keep me from going past step 3 most of the time anyway.
@@SimGunther Hi can you please suggest me some program/system I could write to test different paradigms and even custom rulesets for their tradeoffs and to also upskill myself in different modes of programming/designing systems. TIA
That’s pretty much just called “programming” and how it should always be done.
1) do exactly what you need to do with data.
2) abstract out the common state for compressed and efficient code.
that's a nice analysis
Brian Will 7 years ago made a video on why OOP is not really OOP and ppl still discuss it.
The 'look in less places during side-effect-free FP' is sorta true, in the sense that you don't need to look around for more and more things as you read through a function, the scope only gets tighter and tighter.
Like if you write a django view, at any point the request could hit the DB and influence the state of the app, but with Haskell or whatever, unless you're doing side-effects (IO), you can be sure that you only have access to the variables you did at the start of the request. You don't need to check what state the database is in, because it wasn't passed in as a variable.
Hard to put into words but it does feel simpler (in certain aspects) compared to the imperative style, even if there's tons of overlaps. Its all about picking your poison for sure.
A Django app was my first large production OOP based thing to work on.
It's awful.
As you go into things to figure out what's going on it just goes everywhere.
"the scope only gets tighter and tighter."
This is kinda the essence of composition in OOP. However it's not that simple, it's usually almost impossible to design code in purely hierarchical manner. Now OOP has produced plenty of great literature how to design your application that can achieve this property - onion/hexagonal/etc architectures are all born from applying the principle of composition to large apps.
@@sk-sm9sh Pure functional is all about composition because there's no way not to do it!
Tons of purely hieracrchical projects exist using Elm or Haskell since they don't allow using something you don't pass in (with some tiny exceptions!).
You should try out some FP languages if you're curious!
@@TankorSmash here we go again "you should try some FP". From my experience really more people tried it that FP zealots like to think. Just because I tried it doesn't mean I love it already. To me it falls short in many regards.
@@sk-sm9sh You're saying you tried one of them and didn't like it? That's totally valid too! There's unfortunately no silver bullet after all.
I hate this generation of devs. Everyone and their mom want to be come „dev influencers“ when they are obviously either just BAD at it, just graduated or are just repeating the bad Medium posts of others.
Saying „public static final“ is a OOP pattern is so stupid and then continuing „now I want to show the beauty of FP, but I never used it so sorry I’m advance for my bullshit“ is so ridiculous.
„Engineers“, we are laughing stocks
Unfortunately true.
I sat in disbelief as print("Hello World") was labeled as an example of functional programming
And using print, a function that only has a side effect, as your example for functional programming is just ridiculous. Not to mention that he just writes a wrapper function to print which requires that the argument is string, but passes it directly to the print function which uses dynamic dispatch... The video seems like such a confused mess
I think one of the most overlooked upside of pure functions and immutable variables is the ability to distribute (and cache) their workload. Since you don't have to track a shared state and the output is deterministic all the time it does not matter which machine (or processor, or core) works on your function, the result stays the same.
This is basically the raison d'être of Erlang (and by extension Elixir), it's astonishing how seamless & lightweight process creation is. I'm not aware of any other language where you can be running thousands of threads without even batting an eye.
@@ultru3525 Unison would be another one.
oop has a problem. creates redundant elements in memory. In many programming languages an object inherits from a base object. Therefore, every time it is instantiated, its elements, properties, methods, special methods, etc. are also loaded. fp doesn't do that. The data is separated from the methods, whether they are pure functions (calculation) or impure functions of actions. In addition, the Earlang virtual machine does one thing, and that is the virtualization of processes, so these are not loaded into the operating system with a life cycle and everything else, they are done within a virtual manager of processes and their Manager and process hierarchical trees
4:00 abstraction is such an abstract concept
Not it's not. It just means abstract methods as in eg interfaces.
Functional Programming can't work with side effects?, in Haskell (Advanced Pure Punctional Programming Language) you can work with effects using Functor, Monad and Applicative
Correct. Haskell manages those effects.
The vast majority of FP in the wild goes on in imperative languages. Functional-style code in Python and JS is still just a bunch of procedure calls. That's because these are the mainstream languages.
(except Excel ofc)
But the converse is also achievable. You can take a purely functional language and build stateful, imperative abstractions within it.
Btw, you should really try expanding your view. Try learning some variant of Lisp (Scheme seems solid and small-enough) and something with its roots in ML (maybe Elm, since it focuses on FP novices and is very small too)
Common Lisp is best lisp. Scheme is so small that tons of important functionality is implementation dependent. Common lisp has a big standard, but excellent facilities. Also, unhygienic macros, unwind-protect, CLOS, excellent exception handling, optional dynamic scoping, etc.
Clojure has neat features and excellent java interop, and scheme is super small and really lets the implementor play around the design space. But common lisp's interactive development is top tier, even among lisp's.
Also if your compiler/interpreter cannot collapse several chained map functions into a single loop it is not really functional programming. It is just functional style.
Take SQL for example, you can nest SELECT clauses all you want, but it isn't going to create a bunch of temporary tables if it can evaluate the expression with one index traversal.
SQL in that regard is much more functional than the functional part of JavaScript. Of course in SQL you can't store functions as values in tables, so that is where that ends.
@@colemanroberts1102 Common Lisp definitely is more suitable for making relatively fast real-world applications, but for someone more interested in learning programming language paradigms & fundamentals, Scheme's minimalism & simplicity is actually a benefit, e.g. I couldn't wrap my head around the concept of an "object" until I had to implement them from scratch in Scheme as a CS freshman. Other things we implemented in it are a type system (using cons cells), lazy evaluation (using lambdas), DSLs, a filesystem, a relational database management system, a meta-circular evaluator, and just about every data structure known to mankind. It's the ideal teaching language imo.
Also, Racket is a great scheme variant that offers the best of both, with a comprehensive standard library, package manager, build tool, and top-notch documentation, it even comes with its own IDE, the only thing it really lacks is speed.
@@ultru3525 minus the meta-circular evaluator, all those things are just as easily done using the same techniques in Common Lisp. You are correct about scheme being a better platform for learning to implement a language though.
@@colemanroberts1102 Oh yeah, I wasn't trying to imply you can't do those in Common Lisp, my point was that all of those can be done in Scheme, in spite of its limited facilities.
And as a teaching language, I feel like Common Lisp suffers from the same problem as Java: too many options that are useful in production, but can only serve as a distraction when you're trying to focus on the fundamentals.
Common Lisp was the second language I got comfortable with after Python, and while spending two days trying out all permutations of `loop` keywords was interesting, it wasn't really the best use of my time. With Scheme, it's practically impossible to "waste" time learning language-specific features, you'd have to delve deep into the intricacies of continuations for that.
I bet this Codepersist guy will look back at this video in a few years with embarrassment about all the misconceptions he exposed. But I guess putting together the video taught him something, so it's all part of the journey. But my concern is, how many other junior devs he'll confuse through his public learning journey.
"I already hate everything i see" made me spit out my coffee
Encapsulation is not about data at all. Encapsulation is about implementation hiding -- David Parnas called it information hiding, really meaning hiding the information about the detail of implementation.
Thinking private is about encapsulation is wrong. But that is not the only reason private is wrong. Hiding fields from subclasses is wrong.
People mostly use Information Hiding not to hide information, but to hide the the gross stuff they built when no-one was looking.
04:15 yeah abstraction as a pillar never made sense to me either. They're all abstractions.
The guy in the chat who said I just use global variables, is a menace to society.
Nothing wrong with global variables
The great thing about Python is that if I manually set the recursion limit, I can choose which level of abstraction I live on. I can get most of FORTH's extensibility if I want it, or I can have completely pre-cooked loops if I want them as well.
Yo! Fib or exponential indenting seems like a great idea. Arrow antipattern no more.
LFG!
9:05 he used brackets in python?
"Abstraction is like everything..." Nailed it!
5:20 I find it more senseful for Rust to use Wrappers(sounds pretty similar to encapsulation for me) not to protect sensitive data, but to safeguard the data's/ or type validity.
For example, a non-empty string which is also a valid email. This Data-Type we will call it "Email". now, everytime you use Email as a func's argument, you can be a hundred percent confident that the function is both receiving a valid and non empty e-mail.
And furthermore, when copy-pasting the function (let's say the version that use a String as argument instead of Email), you won't accidentally forget to parse the string, or even do a double/multiple parse(one inside the function, and another at the callsite(s) because you forgot the function would parse the String automatically for you)
Please keep doing what you're doing! I love your content!
Lil bit o' both!
0:55 Java is Multi-paradigm tho (Java has functional programming after Java 8, with streams and labmdas)
Functional programming maps so well to many things in mathematics and Computer Science but OOP maps well to the real world.
For me though, FP is good when I'm building and all the ideas are in my head but if I come back to it three days later I'm confused.
OOP maps well to the real world?! We must be living in different worlds then.
@@etodemerzel2627 ehhhh...? I know you can create custom types and there is the data keyword in Haskell but... I don't use it... maybe that's the world I'm from.
@@etodemerzel2627 On the fundamental level the living world around us is a bunch of cells communicating. Like a bunch of "objects" passing "messages".
@@freesoftwareextremist8119 That's true OOP or Actor model as it's known these days.
It’s a grave error to try and map the “real world” to code. Code exists entirely in 0’s and 1’s. There is no “chair” that inherits from “furniture”. There is only data.
Figuring out how not to have so many copies was one of the challenges of learning FP.
The best paradigm is to not think about it
Thanks!
sorry i didn't say thank you for this!
i just learned about super thanks ;)
oop has a problem. creates redundant elements in memory. In many programming languages an object inherits from a base object. Therefore, every time it is instantiated, its elements, properties, methods, special methods, etc. are also loaded. fp doesn't do that. The data is separated from the methods, whether they are pure functions (calculation) or impure functions of actions. In addition, the Earlang virtual machine does one thing, and that is the virtualization of processes, so these are not loaded into the operating system with a life cycle and everything else, they are done within a virtual manager of processes and their Manager and process hierarchical trees
Good video. Real quick, total noob here, why would you avoid inheritance? It just strings too many things together causing bug enhancing dependencies? It seems like such a cool idea to be able to create your world through a process similar to nature. I've had waking dreams of using a few primal classes and generating a universe via inheritance. I guess it isn't as pretty irl?
This is from a Python point of view. The only reason to use Classes is to simplify a complex object into JSON basically a dictonary. Sometimes it's easier to retrun a dictionary with a Class than a function. Functional programming only to not change the initial state of that object (Immutablity). Procedural programming for everything else. The thing I like about Python is that you can abstract basic things with a context manager. This is like procedural programming without having to worry about the end. For example, writing to a db. I don't have to procedurally say...insert, commit, close, roll back if something went wrong. The context manager handles all of that. Also, If you need to go for speed (concurrency or multi-processing or some combo of the two) it's simple enough to work with if you use a Class to make your iterable and an asyncronous function to blast the hell out of it all while avoiding bugs caused by inheritence/state/polymorphism because you're at least adhering the the principle of immutability and can thus better deal with the expected output or raise an Error/Warning.
Disagree here. FP is an approach. A mindset. FP promotes high cohesion, low coupling, side-effect management, state management, etc. all ideas the React borrows. Unix piping data around is FP. The web's statelessness is FP. XML/HTML is just LISP with syntax sugar. It is FP. Just because you create a class doesn't mean you're not following FP principles. And just because you use Monads like Promise doesn't mean you are.
10:17 this is why js is sometimes so slow: just a bunch of spread o\perators. spreading their filth all over the program. love it! LOVE IT
12:15 you missed calling out the if (bool_expr) return true else return false horror.
This needs to be a short... "What have you dobe here? What have you done? This things like 12 characters here"... And then another short that overlaps a little and includes the Fibonacci bit
Embrace no paradigm: all hail the mighty Nim!
Design paradigms are tools, and become problematic when treated as dogma. It's a take I wish I would see more often.
REACTIVE + DECLARATIVE (language oriented) is best
Would somebody mind explaining what prime meant by properties being one of the big problems with OOP/classes? Im developing a language and I'm currently writing a EBNF/grammar for the language. I think some approaches I have come up with can avoid OOP confusion in some areas but I'm stuck on what prime meant by the properties comment, could be interesting?
I can't work out if the diagram at 4:30 is an inheritance diagram. If so that is an insane hierarchy. Too often people use really bad examples in OO and say OO is at fault. Subclassing is a very tight relationship and should be used carefully and sparingly.
Car --> Toyota --> Truck is not an inheritance hierarchy, what is it? It does not make any sense really on any other level.
People think of inheritance as a mechanism for building up taxonomies, which it is absolutely not. They also get the whole reuse-through-inheritance completely wrong. They think "Oh, I want to reuse that class, so I'm gonna inherit from it." But the point of inheritance is not to inherit functionality, but to implement towards an interface, so that the code that _operates on that interface_ can be reused.
Also: Square aren't rectangles as long as you're not willing to accept squares with differing side lengths.
@@pillmuncher67 I think you are confusing taxonomy and reuse. Inheritance is very much about type taxonomies. I agree that thinking "I want to use/reuse that class, so I must use inheritance" is completely wrong.
However, if we use inheritance for taxonomies, correct reuse will follow. That is use interface inheritance, not implementation inheritance. But Java is completely wrong in separating interfaces from classes. All classes implicitly have an interface. There is no need for an explicit interface concept.
Inheritance constrains the parent class. Thus all squares are absolutely rectangles. The definition of square conforms to rectangle -- a shape with four sides the opposites are parallel and all angles are 90º. But square also conforms to rhombus, a quadrilateral that has all four sides equal. A square can thus be defined by multiple inheritance between rhombus and rectangle which defines square.
Remember inheritance deals the the abstract definition of entities. Now there can be implementation attached, but not fully specified, and this is what is wrong with Java interfaces.
See chapter 24 of:
bertrandmeyer.com/OOSC2/
which discusses correct use of inheritance.
@@ijoyner Yes I know that book, I have it on my shelf. There are points where I agree with Meyer and others where I don't.
But ask yourself: If you have a Rectangle class with a method set_width(n:int) and the post condition is that only the width is changed but not the height - what could possibly be the meaning of this for a square? Can squares have sides of different length? Should an error be raised? Should you write an if statement to test if a rectangle is, in fact, a square? Why not just make a new Square class that inherits from, say, Shape, and that holds a private reference to a Rectangle object to which it then delegates? Like, if the side length has to be changed we call both set_width() and set_height() with the same argument.
One must always look at the problem a specific technique is trying to solve. In the case of OO, as it occurs in the wild, the problem is to have polymorphic method dispatch over one parameter at runtime, so we can add new things without having to go through the code and add case blocks to switch statements. Everything else (entities, taxonomy, ...) is marketing.
@@pillmuncher67 The meaning for square is that both set_width and set_height set the length of the side of the square since the invariant of square is width = height. Perhaps we inherited from a general quadrilateral which has sides l, r, t, b. In that case for square l = r = t = b. Perhaps we should not label the sides left, right, etc since we might rotate the shape.
"Should you write an if statement to test if a rectangle is, in fact, a square?"
No need. That is expressed in the class invariant. That is essential properties of objects are separated from the code.
That is the purpose of redefine, select, and undefine in Chapter 15.
"Everything else (entities, taxonomy, ...) is marketing."
Not at all. Taxonomy is a scientific approach to understanding things, the similarities, the differences, the constraints that distinguish a subclass from parent classes and other subclasses. Taxonomy is almost the most important part of the technique. But taxonomy is not exclusive to OO and can be used in other paradigms like FP. Type taxonomy is important. But it should not be used for reuse first. Let the reuse occur as a pleasant consequence. That is where people go wrong.
These are all considerations and considerations that inheritance brings up. That does not mean it is wrong -- you will face these questions whatever. You could just implement as completely independent entities, but then the relationships between the shapes, the commonalities and differences won't be apparent and no reuse will result.
(And don't use '()' to signal 'function' -- '()' is a primitive call operator and ugly Hungarian notation exposing implementation.)
@@pillmuncher67 I think the other aspect where people go wrong about inheritance and taxonomy is they are thinking that B inherits from A means B has added to the functionality of A. But A expresses many possibilities and B is actually taking away from that.
When people think of implementation inheritance they think that is everything A does, and we add what B does. Well, no, we put the general facilities in A that cover a lot, defaults. The job of B is to subtract from that. Again to express by constraints why B is different to other subclasses of A. That is interface inheritance (but again Java is wrong in separating the notion of interface from class, and that in itself results in very artificial inheritance hierarchies).
Not true about procedural. There is a thin mapping layer you can actually get to. So if you have a clear algebraic structure set up (association and id’s), you can pretty much get away with behavioral and dataflow modeling. But I get why one might perceive this as a procedural approach. The functions are pure, though.
I like DOP (Data oriented programming). Understand your data and the necessary transformations you need for that data. Then your code will solve your problem (and nothing else). Writing it will be straightforward.
I am learning DOP now, but is only popular between game devs now, since it uses very unconventional syntax and is harder to extend.
I have seen languages like Zig, that are trying to make DOP more ergonomic, you can write a conventional struct, then use a function from the stdlib, so it will be converted to a data oriented pattern under the hood.
@@rj7250a any resources you have for DOP ? thanks!
At 5:00, can we talk how this guy animates the code ? It’s magic.
@7:50 i whole heartedly agree, but i think we all do this.
its just more, the leaf nodes, of what you use, should contain all of ur behavior and modifications...
im trying to think about how i love protocols, and typing but hate inheritance... having a protocol is such an easy way to perform a type union, even if u have to define an interface. it prevents messy if logic and gives a path forward, with only 4 more lines of code.
unfortunately, today, i found myself liking asserts as a way to force typing, when underneath an optional, when unfortunately i needed an optional, as i have a conclude method which creates the return object, but theres a step which only executes for some of the paths.
turning on mypy for the typechecker shozwed me how much happy path cofing i was doing 😮😅
I drink your programming paradigm. I drink it up!
In the end, the good ole structured programming is still what works best
I think the intention of Functional Programming is for solving problems in a mathematical way. Like you said mainstream functional programming is just organizing your code in procedures without side effects. Same with OOP where Alan Kay was like it's more about the messages than the objects themselves, but objects are just another way of organizing your procedures. Procedures FTW!
"Address and SSN are sensitive data therefore they're private, but name isn't so it can be public"
That's…not the goal… It's a terrible misunderstanding of the point of encapsulation!
Correct me if I'm wrong but to me encapsulation is used to better monitor the use/change of variable and avoid messy classes using eachother variables directly without a going through a function you can use to understand the situation better if, for example, you're debugging code
@@Malenbolai Basically the goal is that everything outside of a class should not be concerned about its inner workings. That ensures that the data within is always treated consistently and according to the desired logic. So yeah, it's pretty close to what you said.
It's not about what the end user can do, it's about what the _developer_ can do.
@Wolfeur thx for answering and the added insight pan
abstraction was more of an obstruction to me
Alan Key's OOP is not mutually exclusive with FP. I write basically the same code in C++ and in Haskell, just with different syntax.
I love when prime freaks out over spaces and indents
ff vs oop vs pp really depends on what you're making and how it should read when you're debugging it. IMO it should be natural. If you're hesitating on paradigms, patterns, syntax, etc. you need way more practice.
If you need hierarchy use OOP, if there are clear steps, make it obvious, and I guess otherwise try to preserve immutability at all costs.
Yay! Finally someone calling out code for its bad indenting. F'r f's sake! The programmer should not actively be trying to deceive the reader.
Procedural Programming Andy is a title I might be down with for a time. But I would defy it on occasion.
2:37 Crazy default inheritance system in Python is precisely what made me avoid OOP in Python and almost made me quit Python as a programming language.
*BUT*
Multiple conditional inheritance in Lua, explicitly implemented in the program by using the Lua language itself and exactly according to my special needs, together with memoization, once made me write an elegant, fast and memory-friendly pool of procedurally-generated sprites for real-time visual effects for a video game. Because I designed and implemented the inheritance mechanisms myself, I used it as intended and wasn't struggling with some bad generic compromise made by language designers. *I made up my own OOP mechanism according to the type of problem I had to solve, and not the other way around.*
Building your own inheritance system is very easy in some langages, requiring no special keywords nor classes, simply by using the available structures in order to define closures and conditional calls, and let your decide exactly what is inherited, under what conditions, and how exactly (priorities, copies, references, memoization…), even at runtime.
But, yeah, what I have learned from this is: *use FPL to define OOP-like multiple - or even concurrent - inheritance as you like* , but *do not try to practice FP with a rigid OOPL* unless you plan to hate yourself.
If he sees this, to fix the screen tear you can use the picom composite manager, just turn off window in and out fading shit. Worked for me
seeing this video 7 months later and knowing that he hasn't moved to wayland makes me think problably 2024 is wayland's year, as 2023, and as 2022, and as 2021...
I'm not convinced that the person in the video (not Primeagen) actually knows what he's talking about. Properly formatting the example code would improve the appearance of reliability.
I'm a big fan of FP, but I do agree with the general idea that multiparadigm is probably the most optimal way. Engineers in all disciplines are taught to "pick the right tool for the job", but software engineers don't seem to take this to heart and they end up being hellbent on proving one approach right. You're taught to break down your complex problem into a bunch of simpler ones, and picking a paradigm is the same deal: some of these sub-problems will fit one paradigm better than the others. Procedural rules tho'
I heard an interesting take on this whole paradigm war:
- Use object-oriented ideas (encapsulation, representation hiding) for your core data
- Use functional programming (immutability and pure functions) for your business logic
- Use procedural programming for the tasks that talk to external systems
I might have forgotten a paradigm or two, but I got the central message right: use the central ideas of each paradigm in those parts of your code where those ideas accomplish some real, useful benefit.
Also, OO people talk about data hiding. That is wrong: getDayOfMonth will return data. It cannot be hidden and it should not be hidden; it is the essential point of a Date object. What _can_ and probably _should_ be hidden is how this value is arrived at, and how this small integer is represented internally. Likewise, the exact representation of nodes in a red-black (or AVL, or B) tree is probably not something the user should know about; they should just have a Map interface and a logN performance guarantee provided the comparison function is sane. But the point is not to hide data, the point is to hide how data is represented.
TL;DR: it's _representation_ hiding, not data hiding.
> Also, OO people talk about data hiding
What people talk by data hiding is exactly what you mention later - hiding internal representation, thus establishing encapsulation. Nobody has ever said that you should not have methods that return values.
Data Oriented design + manual memory management + imperative code with no hidden control flow, or GTFO
Ansi C
I strongly object to distinguishing OOP/FP on auto-completion. Packages export constants, functions, types, and even function-types (if you export mutable values, you gotta go see Karen, my man), where class instances export mutable values, and bound-functions (methods).
It’s the same thing, with different flavors. It’s just namespaces, at the end of the day.
What people who understand the issues are talking about when debating OOP/FP is whether the ideals of either are worth the tropes. You can write shallow OOP, preferring composition over inheritance. You can also write OOP with deep caverns of multiple inheritance, bottomless pits of overrides and monkey patching, and always the subaural drumbeats of doom beating out “Meta..classes…meta…classes…meta…classes…”. FP is more dependent on the compiler to do well, but Rust’s iterators are a FANTASTIC example of a functional paradigm that has great support from the compiler, is fast, and is still very clear and modular.
Honestly the biggest thing for me: write the code you need to get the job done, keep it clean and responsible, pick up your errors like a decent human being, and check for null/nil before dereferencing. It won’t matter if you don’t have setters and getters on every public property or if you mutate data in place or return an altered copy. What matters is that future you and friends will be able to observe what is changing and where and why.
So yeah. Hardcore object to insisting a pattern is too hard or unreasonable without understanding it or where the payoff is. That said, copying every value in your array to return a modified copy - or maybe even not - can obviously be an expensive price to pay for making your code more readable.
Overall, I do agree, but maybe I’m just…not a developer? (I worked as a dev in the past, though)
You see, for me, classes of equivalence matter a lot. So to keep track of that, you must follow stricter rules. Pure functions definitely help with that! Too much perfectionism..sorry. I don’t like (regular) dev work. Waay to much mambo-jumbo and spaghetti.. I’m more conceptual..😢 I suffer..
@simoninkin9090 , I think I'm saying that, but with more mumbo-jumbo. I'm pointing out that there are some real considerations, but if all you care about is being able to put a name followed by a dot, and have your IDE suggest completions, you can do that in either paradigm, not just by using classes.
@@samhughes1747 I totally agree about the necessity of good namespacing. The question is ultimately about composition. Sure, objects can also compose (I did some of that a few times), but then these objects must comply with the similar rules as pure functions. And it is actually harder to work with.. you might want to call these “function objects”. There is an override in cpp for such cases.
You can definitely do everything in either paradigm. In fact, as a _professional_ (the one I do not ever want to become…because screw the payoff! =)) ) you should be able to slide though the codebase, written by other people and follow the paradigms in the code itself. All significant modifications must be accepted by the tech lead, etc..
What I’m talking about is kinda R&D…or whatever. Maybe it’s not that practical in real life (though, the finished code can be optimized and quite fast..just that nobody would understand it..that’s all..), but I honestly don’t care. There is no freedom in commercial programming. For some people it’s..well - just a job. (Don’t get me wrong though, I’m quite average). Yet, for me, it’s more like a tool for understanding deeper realms of our universe. Not freakin’ kidding! =))) FP maps very well onto many concepts in modern physics (I’m not a physicist, but I’m really curious and can’t stop digging for answers..or rather - more questions)
P.s. never actually touched Rust myself, but the iterator pattern sounds like it’s the same as STL iterators. In fact, the essence of C++, the Standard Template Library is also functional! But it’s a mess! If you compare that even to F# (hate dotnet though..or any vm, which is not llvm. Bytecode really sucks!) or even better - Elixir (possibly the cleanest syntax I’ve seen in a programming language)
@@simoninkin9090, gotcha. I was mostly complaining about the idea of dismissing FP based on auto-completion advantages, and emphasizing that OOP and FP both have namespaces that lend to that auto-completion. It’s one of the most common complaints about FP I hear, and it’s nonsense.
Meanwhile, yeah, I work on a team with a variety of skill and discipline levels. We have a lot of code that drags a logger of some kind through the whole program, and same for whatever database or API client is being used. It’s nauseating, but I’d the program does what it’s supposed to, and if there’s tests to PROVE that pile of tech-debt does what it’s supposed to, I’ll sign off.
I care far more about my coworkers doing good failure-mode analysis and proper error-handling. My team lead will back me up on complaints about squashed errors or untyped interfaces, but he isn’t going to back me up on holding work hostage to a paradigm that is generally widely misunderstood.
I absolutely am convinced by the ideals of the FP paradigm. In practice, like you say, there’s everyone you work with, and you’re writing code for them to read too.
The name... is the Procedureagan
I think classes are ok way to abstract the different designs but the issue is when people teach these abstractions where you have to drive it from some real life concept and that doesn't work with software because you have to understand the context where your application is running. You abstract the application to memory etc and not some stupid animal that becomes ketchup.
3:15 Somebody explain to me why he's shouting that.
Wasn't this Terry Davis' take years ago minus the rants about running over glowies with a car?
7:50 What I hate most about inheritance is being able to override things multiple times.
Something should either be abstract or maybe virtual with a default implementation, and you only override it once.
Else something like this could happen.
There's a class A. Class B inherits class A, since it's basically the same, but almost everything is slightly different.
Class C inherits from class B, since it's basically the same, but almost everything is slightly different.
Why not just let class C already inherit from class A and duplicate the one method from class B, which is still the same?
Or even better, turn this method into a function.
Maybe just create an abstract super class for A, B and C, all three of them, and only make the parts, which are not exactly the same abstract methods?
"leave a bible width's space between the two of you!!"
If I see another car example used in a discussion of OOP, I swear to God...
5:30 the name,, is “7layers_of_inheritance_is_the_only_way_to_program_agen”
Bro seriously said he prefers programming in spreadsheets than rust.
Just write lisp and lose all ability to distinguish between the two consepts. Because the functions don’t share the same namespace with values - symbol can have both. And functions are valid values as well!
I think picom has a 'vsync' option which gets rid of screen tearing.
12:50 Yeah I like C# extension methods for this reason. print(derpNamespace.derpFormat("Surprise MF!"));
becomes print("Surprise MF!".derpFormat()); and you don't need to remember which file/namespace derpFormat is in
13:03 return False outside function???
I’ve actually stopped using encapsulation, at least in the explicit, language supported sense. Now I essentially make everything public and static, and I simply use organization and architecture to implicitly enforce simple rules about what code is allowed to call specific methods or access specific data. I like to think of it as Extreme Domain Development.
Implicit rules are ok if your team is small. How you're going to enforce it if there is 100+ people working on project? Java has so many access control keywords for large teams and indeed they are unnecessary for small teams but for large organizations being able to specify access-control is very helpful as it's makes enforcing ownership easier. For instance I tell that I own this class, and I put everything as private. Now it is easily guaranteed that no one will access the private vars without modifying the file. Since I observe changes to my file I can easily notice if someone tries to get stuff that I marked as private.
@@sk-sm9sh Having 100+ people working on a single codebase is becoming pretty rare these days. I only work on two types of projects right now - personal projects with one developer, or micro services with 1-3 developers.
But, hypothetically, if you have more developers, you don’t want them checking in code that hasn’t been reviewed by at least one senior developer. At that point, it would be their job to ensure that the code is in the right file, and that it either has zero dependencies, or that it only references code that is located in its sub-domain.
Think of it as an org chart. If everything is organized properly, you don’t normally have salespeople talking directly to developers. Technically, you can, and it’s not really the end of the world if it happens. But ideally, you want that information to go through some sort of management structure, so that resources can be prioritized properly.
I essentially do the same thing with software architecture. I have components that actually do work (read files, send messages, query data, etc.) but make no decisions, and I have “management” components that wait for event notifications from sub-components, and then send commands to other sub-components. I also try to keep the number of sub-components managed by a single management component to a minimum, usually no more than 2.
This makes design work fairly trivial, and when something doesn’t work properly, it’s pretty obvious where the problem is. Although I’d have to say that this approach really does make things so simple that code often works as expected on the very first try.
llvm api is written in c++, so by that logic, rust is also a c++ abstraction.
It doesn’t really matter if you up the abstraction level, or go low level. At some point you’re gonna converge to a pure categorical realm. So yes, you are correct! Everything is just a category!
I've recently faced the same screen tearing issue on my desktop (nvidia rtx 3070). Arch linux wiki suggested to add a "metamodes" option on "Screen" enabling NVIDIA "ForceFullCompositionPipeline". The option did fix the issue but it also made the refresh rate feels slow (feels because it was still 60Hz but it was looking like it was 30Hz). In order to fix that, I've had to allow non-edid modes by adding the Option "ModeValidation" "AllowNonEdidModes" in the "Device" section, and I've had to set the resolution to an XServer resolution and not use the EDID suggested one.
That was quite tricky, in the "Device" section I added an Option "ModeDebug" "true", read the /var/log/Xorg.log, and I've noticed there were 2 mode entries for the resolution 3840x2160: "3840x2160_60" and "3840x2160_60_0", the former was from EDID and the latter from XServer.
The final config looked like this:
Section "Device"
...
Option "ModeValidation" "AllowNonEdidModes"
EndSection
Section "Screen"
...
Option "metamodes" "3840x2160_60_0 +0+0 { ForceFullCompositionPipeline = On }"
EndSection
As my sight is no longer great, I've also set the DPI to 192 by adding the following line to my .Xresources file:
Xft.dpi: 192
I hope that it helps you (or anyone else)
All these languages ages are about handling complexity. Whenever paradigm is simple and accessible enough for people to take advantage of it, that’s what matters.
Most of my JS code these days is based upon async iterators. Stack a bunch of async generators on top of each other, pull an item from the bottom, and the generators yield all the way down. Each generator is self-contained and a mix of procedural and declarative. Debugging is easy.
Sounds terrible lol
@@sk-sm9sh Thanks!
What video editor the guy is using ? I mean the author of the video
What's your opinion about ramda ?
if your code isn't well formatted, that almost immediately throws your credibility out the door for me.
Noob question, how does declarative fall between this? I assumed declarative was functional
3:31 of these inheritance is typically a footgun and strongly advised against, and encapsulation is a way to bloat your source files 8:1
Encapsulation isn't just getters and setters
@@zachb1706 my ratio was optimistic then?
The point of encapsulation is to avoid your code from becoming spaghetti where everything depends on everything. Getters and setters allow that to happen again.
@@falsemcnuggethope yes because then you don't have to call setters to set the value of values anymore.
the return type of getters depends on the type of the field. and if that ever changes, now you have to change both the getter and everywhere you used it. it changes nothing.
or perhaps i am misunderstanding your argument. how exactly do getters reduce coupling?
@@SArthur221 having only a getter enforces that the dependency is a one-way pasta string, but it's still a pasta string. If you have too many of wrong ones like that, you end up with spaghetti, just as you do if you access member variables directly.
That's why we'll all go for Data Orientated Programming ;)
DOD: If premature optimization were an entire paradigm, lol.
@@zacharychristy8928 "Premature optimization". I suggest you look up the full phrase and the context... This phrase is basically invalid these days. Please, stop using it.
@@etodemerzel2627 Nope! Still totally valid. Optimizing your solution almost necessarily makes it less general. Doing so before you have a complete picture of the problem and requirements is at best a waste of time, at worst a disastrous foundation to build your program on.
Your choice of paradigm should be shaped like the problem itself, and when the problem isn't easily modeled by simply abstracting it as "data and transformations" (which describes plenty of problems) then DOD isn't a great fit.
@@zacharychristy8928
So, I agree with you that the notion of "premature optimization" is still completely valid (actually, it just doesn't make sense to say that it's "invalid", it's not a timing problem). But at the same time I think your statement is complicated, all computing problems can be boiled down to "data and transformations", literally all of them. Data-oriented programming is an excellent way to solve many architectural problems while also solving performance problems (of course, it doesn't solve all or is ideal for all types of problems, but overall it can solve most of them efficiently and cleanly).
@@zacharychristy8928 Aren't all computing problems basically "Data and Transformations"?
Also I do not see the problem will less generalized solutions. Indeed most solutions should be completely specialized in my opinion.
Either the video is made by an amateur or it's supposed to be a mockery of another video... made by an amateur. As we say back home "if you'd have shut up, you'd have remained a philosopher".
Why on Earth would anyone use multiple inheritance?
It makes debugging a lot harder.
Instead of going up a linked list of parent classes when looking for a bug, you have to do a tree traversal over a branching structure of parents.
Which would be tolerable if debugging was done by machines. But humans are barely attentive enough to properly follow a linked list while debugging. The chance that a developer will traverse a tree of classes in an efficient manner is more or less zero percent.
Is there some use-case I'm not aware of where interfaces would work worse than multiple inheritance?
I've used it once, to splice together a status class from multiple partial status classes in C++. It worked very well, but those were also glorified structs.
If you have to write the same line of code twice its ok, but if you do it three times you're playing with yourself.
This could be extremely elitist but I don't trust someone's videos if they mess up their indenting and code so blatantly. Like I even doubt that they code at this point. There's just no way.