Don't know if you're being sarcastic. That is precisely why static types exist and why I very strongly dislike dynamic and weakly typed languages. You have to think too much about what might go wrong. You already have to do that plenty.
I cannot take credit for this since it indeed is the very idea of static typing. Nevertheless I find it very important to keep reminding ourselves of that 😊😊 Thank you both for watching and sharing your thoughts 😊🙏
@@_iPilotexcept that there are too many ways to escape the static typing in typescript. It's an improvement over js, but still ultimately just papering over the gaping holes in js's type system.
Dude, I just want to say thank you. Many years ago I was learning design patterns from you, and now, fast forward, I am great senior dev. My dream came true. Thank ypu for being part of my journey.
another annoying thing is that Array implements IList in C#, but when you call IList.Add it throws an exception, so you can't be sure you can safely call .Add on a method accepting an IList
You’ve earned a new subscriber. Your explanation style is so unique and satisfying to watch, I think in mind the way you emphasize facts, and that’s why your teaching style is relatable to me.
This is a brilliant explanation of LSP ! I love C#, it is a fantastic programming language, but as every programming language there are always some quirks due to historical reasons. I have heard about this "issue" before, but never explained at this level of clarity. Please keep posting these videos, preferably also using C#
Man I love you. I see that you're back on TH-cam after years. I have encountered your channel after university bullied me into Software Design, but you made it slightly more tolerable. Anyways, thank you so much. I will be having my exam in two days. If I pass, it's totally because of you. Even mentioned you in the Course Review section of my course, as our course was probably the most boring thing I have ever seen. However, you made me see the beauty of it, even though I totally hated it all. You just earned yourself a permanent member of your community :)
@chrisropher you should define the Okhravi principle: "never solve at runtime a problem that can be solved at compile time" 😊 By the way this is one of the best video I have seen on software development!!
Thank you. Those are some very kind words 😊🙏 I’m happy that the content is useful. I can’t unfortunately claim ownership of the idea since it’s very old but I will definitely steal that wording and use it. Thank you 😊🙏 and thank you for watching 😊😊
To be fair to the .NET maintainers, certain types have been added later than others and they didn't want to break backwards compatibility. For example IReadOnlyCollection and IReadOnlyList were added in .NET Framework 4. Unfortunately, ROC has always been the weird guy.
As you said, Add throwing when IsReadOnly returns true is not a violation of LSP per se. But there is a violation of LSP for ISet that also specialize ICollection. Given an empty list of integer, when you add 1 twice then the collection has a Count of 2. Given an empty set of integer, when you add 1 twice then the collection has a Count of 1. Had the signature of the method been bool Add(T item), like there is bool Remove(T item), it would have been a completly different story.
Again, that's not really a violation but a very specific, arguably confusing behavior of the Add method. You can call Add and it works, that's all you need to not violate LSP. Also, some degree of freedom is required when we are implementing an interface, not every derived class has to Add to increase the Count. In your example, Set allowing Add on existing keys is actually the better approach, otherwise your Set class would be throwing exceptions which ICollection doesn't define. Actually, Set being an ICollection is a rather good example of trade offs in Software Engineering. By not being overly strict with how Add works, we are utilizing EVERYTHING else IEnumerable and ICollection interface gives us, and that's a lot more useful then defining a separate interface for Sets just to make it perfect.
@@ShubhamKhara I beg to differ. This is the definition of Add. It "Adds an item to the ICollection", so you expect wether the operation to fail (i.e. throw) or to succeed, meaning the Count increases. For ISet, Microsoft had to create a "new" Add method, returning true when the value is added or false when the value was already present. With this signature, the test for the contract is if Add returns true then the Count increased by 1. Regardless of the return value, calling Contains with the value must return true. This is not a trade off, this is an oversight. The trade off is that we have to live with it.
Yes, that makes sense. I had my tunnel vision set on ICollection's function's signature and not ISet's when I initially made the argument. Thank you for the revert, appreciate it.
Man what a great explanation of the LSP. To me one of the most complex principles to understand. Thanks for the video and for the great job explaining it in a way that makes sense.
wikipedia says "objects of a superclass should be replaceable with objects of its subclasses without affecting the correctness of the program" which is pretty clear to me
It seems so complex, because there are a lot of theoretical terms and definitions to make it logically strict. But the meaning is quite simple: Obey subtyping. A square must always be a rectangle. Not every rectangle must be a square. And then abide by that in all conditions and objects of the class or interface you are designing. It's also basically helpful advice to developers to adhere to the subtyping contract: - Don't forget method parameter types and return types - Don't forget exceptions - Don't forget input value checks and return value range restrictions - Don't forget value restrictions on instance variables and properties - both statically, and over time from state change to state change Of course it has applications in theoretical CS - in order to make any logical deductions, you need a strict definition. But in terms of application, it shows you what to look out for, when designing inheritance/subtype hierarchies of any kind.
But... but... what even is the purpose of implementing IList in ReadOnlyCollection? By what logic does one _want_ a "read-only collection" to necessarily also be a "list"? I don't get what the objective of that decision even was.
You can use all the other practical List methods on the ReadOnlyCollection. Plus it makes it easier to use a ReadOnlyCollection in methods or interfaces that defines the use of a IList.
For interface API design in general: If mutation is involved, interfaces should be split up into the reading part and the writing part (also works well with co/contra variance): IReadCollection and IWriteCollection. There are barely any use cases for write-only collections so that design isn't natural to most people.
Why not make ICollection just behave essentially the same as IReadOnlyCollection (and inherit directly from IEnumerable), but then have IWritableCollection inherit from IReadOnlyCollection and add just the stuff for writing to it?
Collections are always a pain in the ass to implement, but I totally agree with you ! Java has pretty much the same problems with their "Collections.unmodifiableXXX" (List / Set / Map / SortedMap...) and they are also using exceptions. Next step: how would you implement it ? I guess using composition, either containing another container inside the ReadOnlyCollection. Or by splitting collection interfaces into "permission accesses", ReadableCollection (get / iterate) WritableCollection (add / remove / clear) ? Or both ?
Thanks for a great explanation as always. My takeaways from this are: 1. A statically typed language gives you the capabilities to design robust software. But a type checker can not do all the work for you. It hands the responsibility to the engineer to make careful design decisions. 2. It was nice to see how applying LSP leverages the power of the compiler and makes the program safer at compile time. Lots of issues could be avoided if we crafted our systems with this in mind.
Your channel is wonderful. Giving out hidden gems as insights for whoever is patient enough to hear them. Do you plan on ever continuing the design pattern series ?
Brilliant video, I love it all! This plot twist near the end made me smile. However, I feel like it's missing the explanation of Microsoft's decision about this move. As you said, there are incredible intelligent people working on .NET design, so that I am sure there gotta be a good reason for that
Thank you so much for this! I was thinking of mentioning it in your previous LSP video but I couldn't recall off hand where it was I ran into this. I even recall leaving a comment on an SO article that dug into it. I ran into this exact issue with either something from work or a side project I was working on explaining collection types. Anyways, thanks!
The stupid thing is Java is even worse at this. They didn't even bother adding a readonly hierarchy. They simply added factory methods that create readonly collections of type (I)Collection
This also founded on dart language by google . Theres List Type called : UnmodifiedList ... and if u call Add item on it it throws an exception same as .Net and i know it violating LSV Principal....but i also think dart is a new language why they make it like this
ReadOnlyCollection implements IList which is an interface for a mutable list. This means it throws errors if you call add() or remove(). MSFT added an “isReadOnly” flag in ICollection which allows add()/remove() to throw errors. It’s messy workaround for a LSP design error and weakens the compiler.
That's one of the things I like in Objective-C. It's main classes are immutable: NSArray, NSDictionary; they have subclasses NSMutableArray and NSMutableDictionary. Though I don't know .NET, I think that the hierarchy of interfaces would have been nicer if the base interface IEnumerable would have a subinterface ICollection (without add/remove), which would have two subinterfaces: IMutableCollection (with add/remove) and IList (with get(index), but without set(index, item)). The interface IMutableList would have two superinterfaces: IMutableCollection and IList and would add the set(index, item). No need for "unsupported operation" exception and you can immediately see what the class actually supports by its list of interfaces.
Another good reason to move to F#. No LSP issues (unless you seek them out). All functional datatypes are immutable by default (which implies readonly). It's so much simpler and more concise, once you made the paradigm shift. Rarely try/catch or exceptions and aprogram you can reason about.
Until you have infinite amount of resources and they are almost free. In the real world, all computations have their cost in CPU time, memory, and amount of data transferred over the network.
No, C# is a very good language, and it's far easier to understand than F#. This critique has a good point, but without showing us the entire design of an alternative, we can't even say it was the wrong choice.
IEnumerable, ICollection, and IList were added to .NET in version 2.0. IReadOnlyCollection was added to .NET in version 4.5. Is the inheritance hierarchy "wrong"? Yes. But doing it "right" would have broken the world. If you value ideological purity more highly than backwards compatibility, perhaps C# and .NET aren't for you. 😁
Inserting an interface into an inheritance chain wouldn't break anything. I.e they could have made ICollection inherit from IReadOnlyCollection. The big problem though, is that the class ReadOnlyCollection was introduced in .NET Framework 2.0, long before the interface IReadOnlyCollection in 4.5. So at the time they only really had IEnumerable that it could sensibly implement. They decided to implement ICollection and IList too, and those can't be removed without breaking backwards compatibility. (I didn't bother with all the s. Too annoying to type on a phone.)
Without a doubt, the ReadOnlyCollection inheriting from IList is a mistake even though the interface is technically correct by throwing that exception. One thing I'd say in addition is, I hope no one thinks that technical validity means the correct approach to using IList.Add is to check if it's safe or otherwise handle the exception in the case of ReadOnlyCollection - we should still code with the assumption IList.Add is always safe to call, and if a client passes in a ReadOnly collection to that method, that's on them to fix.
Totally true. This happens not only with base classes/interfaces, but I have many times encountered such dead-ends in my own code when programming, and the solution is like that one (runtime and ugly). I really would like to see how to fix that.
Inheritance/interfaces are fantastic, the only big problem, for me, is the diamond problem. For example, what if you want to support passing IList to a function that expects a IReadOnlyCollection, because you want to ensure you don't to change the collection in the function that uses it. Under the current hirachy you simply can't do that, unless you go to just an IEnumerable. How would you solve that?
Usually people end up with spaghetti code when they get things wrong. I do remember a former colleague that always wanted to use everything he knew about object oriented programming and his code was horrible. You could easily remove 50%
Speaking from memory (too lazy to test), but I think you can pass the IList straight through without any issues, and it would automatically get cast to the IReadOnlyCollection expected by the function. (IList implements IReadOnlyCollection, so the cast is valid)
Love it. I am now changing my workshop session on LSP to include this. Basically the way that C# has got round it is by a hack and use and a boolean to the ICollection contract. If you need to check the result of this first then does this go against ‘tell don’t ask’? Not a SOLID principle obviously but a commonly used rule.
So basically, the LSP is kept, because ICollection was already broken, so ReadOnlyCollection can be just as broken. Oh the irony. It reminds me of the mathematical property that you can deduce anything from a false statement. Although this obviously has nothing in common with that, it makes sense - if you have chaos, you can deduce more chaos from it...
Wrote my comment (about Objective-C), before seeing yours about Swift. Yep, there are things, which Apple clearly did better. P.S.: Don't like Swift (I prefer Java nowadays), but it's definitely better than mixing C and Smalltalk. :)
i don't really get the point of creating a read-only version of any type. If you want to make an instance of a type read-only you could use the const keyword and mark the methods which don't change the state of the object also with const. Or is the const keyword not available in C# (I'm coming from C++ world)?
My perspective is that it helps to reduce unintended side effects. Say I created function ProcessData(IList list). Since the list is passed by reference, there's nothing stopping me from modifying the list in ways that someone calling the function might not expect. By contrast, if I have a ProcessData(IReadOnlyList list), then that communicates a guarantee (ignoring reflection / recasts) to the caller that the list will not be modified. Side effects can be VERY tricky to debug and hunt down, so imho, it's generally a good idea to reduce those opportunities as much as possible.
Your explanation was engaging and thought-provoking. Made me wonder about the relationship between number of layers of abstraction and probability of violating the Liskov Substitution Principle. For "real-world" solutions of sufficiently complex problems, is this curve likely exponential, linear, or ...?
Hi chris nice video, I have a question that might be a bit off with the topic of this video, the thing is that i started to read a book called "Dependency Injection: Principles, Practices and Patterns" in wich the author explains the different techniques we can use in order to inject properties into our clases (Constructor injection, Method injection and Property injection), he also explained some anti patterns related to dependency injection (Control freak, service locator, ambient context and constrained constructor), all off these stuff its explained on the chapter 4, 5 and 6 and i think you could create a great video explaining thoso topics
@ChristopherOkhravi Can you please explain what could be the possible reason of Implementing IList by ReadOnlyCollection base class. I didn't get this part
IReadOnly** was added much later - after they had made that big blunder with `IsReadOnly`. It's also because (and this might change in .Net 10) IList and ICollection does not implement their IReadOnly counterpart.
this could be an interesting case for type-families (ts ternary type, or rust traits). In fact, typescript has a utility type that does exactly that. i don't think it will be very long until c# gets functions at the type level.
At first glance I'd like to turn the IEnumerable, ICollection and IList upside down from what you did draw. Mostly because I see the IEnumerable as the root and that it then grows upward. Arrow directions can also be confusing - is the arrows the way the functionality evolves or the way the drill-down occurs. It comes down to the way you think, and it's easy to get locked into one pattern (information flow for example) and if someone else uses a different pattern then it takes time to adapt. At least you did set a baseline in the first 2 minutes. As for the inheritance issue I'd like to change so that the indexing that IList provides should be provided by IEnumerable and the IReadOnlyCollection should be between IEnumerable and ICollection.
.NET concrete collections have always been a mess. Collection implements List, SortedList implements IDictionary and those are the ones just off the top of my head.
In Swift, "MutableCollection" inherits from "Collection". Seems like .Net could solve some conceptual problems if they did the same. Well, "IMutableCollection", I suppose, since MS seems to think the name defines what something is as opposed to just looking at the AST and checking if it's an interface or a class. Remindsof them thinking i can change a file from Word to a PDF just by renaming it.
I saw one example of LSP violation myself in .Net source code (when of the methods of base class was not appropriate for the subclass and it was just throwing exception)
I think you were right first time. I don't know why they made its return value to be void, could have just returned 0 or maybe -1 on add() then. At least Java uses a boolean.
Very nice video as always, it definitely was food for thoughts. Letting alone the discomfort I would have using an object of a class that could be (in theory) both mutable and immutable and letting alone the cognitive overhead in terms of counterintuitive naming, I could not help but thinking that this way of deciding the behavior of an object of a given class at runtime with an if-else block may be a little bit of an anti-pattern in a language that allows you to use types (at compile time) to address the issue. What I am wondering now is which reason(s) they may have had to make this particular design choice. Any idea?
Christopher, awesome presentation, with the gestures you remind me a little bit of Morshu :) I want to ask a question because even chat bots confure this one. Is there a circular reference between string and object's ToString? String inherits from object, but object cannot be compiled without string. What do you think?
Regarding the issue in the video, I think that F# solved this issue and applied a much more pragmatic definition of interfaces. There wasn't a need for interfaces, rather the type must have a method that matches the one that is called in order to compile, without the need of interfaces, making implementations very specific and at the same time, very generic, but I might be wrong, haven't checked.
I mean why/when would you require a read-only collection anyway? If you're going to only read through a collection, you don't really care if it's read-only or not, and limiting it to only read-only collections through the type-system would do more harm than good. I think it's just a hard problem to solve, if not impossible, would love to hear a better solution though if anyone has it. Just as a side note, Java solves this in a similar way: UnmodifiableCollection implements Collection.
One common use case would be to communicate and enforce a contract: my library function takes a collection, does some computation using its elements and returns some result to you. In the contract of this function, I promise not to alter any collection you pass me. This is important, because you can now pass me a collection without having to worry (and verify) whether it has changed after I'm done with it. If we have interfaces defining read-only collections, I can both document and enforce this contract succinctly and reliably in one place. Conversely, as it currently stands in C# and Java, you basically have to both read the (prose) documentation of my function to find out this part of the contract and trust my library to actually adhere to it. This makes the contract at the same time weaker _and_ harder to discover, neither of which is a desirable quality in this context. In short, I may not care much if a collection you pass me also supports modification if I only need to read it, but you should care whether I can modify a collection you pass me or not. I'm curious what you imagine the harm done to be of defining a ReadableCollection interface that is extended by a MutableCollection interface which simply adds the add and remove methods. If my function must somehow mutate your collection in order to fulfill its goals, it can define a parameter of type MutableCollection, which also implicitly communicates to you that I intend to alter your collection. If you already have a MutableCollection but my function only requires a ReadableCollection (and defines its parameter accordingly), you can simply pass me your mutable collection with the reassurance that I still can't modify it, without any additional effort on your part. As to the hard problem to solve, maybe you can elaborate on what exactly that problem is?
@@silberwolfSR71 Don't you see the problem with what you're saying? "...defining a ReadableCollection interface that is extended by a MutableCollection interface". If so, the LSP is "violated" once again, and in a WAY worse way, IMO.
@@adambickford8720 Okay but C# isn't a functional language? The LSP doesn't refer to functional programming either. I don't really see your point. EDIT: just to clarify, I see the purpose of immutable collections, of course. But in a hierarchy structure like it's defined in C# (IEnumerable, ICollection, IList, etc), I don't see the point of having an LSP-compliant read-only collection.
So could the developer toggle ICollection.readonly to true or false dynamically so at times the collection is readonly? That would explain the exception at runtime.
IsReadOnly is usually a read-only property. ReadOnlyCollection implementation always returns true. List implementation always returns false. A Collection, when instantiated with a list parameter, just returns the IsReadOnly value of that list. But you could implement it as read-write in your own class if you need to.
Also, arrays are collections that cannot be added to, but can be changed by index. Readonly collections cannot be changed by index. The implementations of ICollection and IList are basically flipped. It would have been better concentrate the mutability methods and properties in the most top level type. IReadOnlyCollections stems from this historic mistake, with ReadOnlyCollection being a simple wrapper around every other collection type to make it work. You wouldn't use it in any declaration.
Thank you I really enjoyed the video, especially the inreractive parts! I guess how I would try to solve it: we would want to introduce an iindexer interface and let IList implement that, then also let the ReadyOnlyCollection implement that interface to "solve" this? (With the information that is used in the video)
My sincere apologies but the answer is unfortunately no. I recorded everything but it didn't turn out as good as I wanted it to. I've got a few things in the works however so it is possible that I might redo it from scratch but that's far out into the future. Again, my sincere apologies.
I think there are 2 ways of thinking about read-only collections. On the one hand, these are generalizations of read-write collections, which only support reading. On the other hand, these are collections which guarantee they won't change. I believe MS devs have change their mind about it couple of times and here's the result
This could be resolved easily with Haskell's static typing. Pretty trivially, actually, so that violators fail at compile time. an isReadOnly() function is a bad idea. The read-only quality should be embodied by the typing directly.
You didn't suggest a fix. Inheritance among interfaces appears to be bad design. When I have something that implements IList, all I really care about is the extra stuff it adds on top of its parent class, so it shouldn't have a parent class. Something that today implements IList, should tomorrow implement IList, ICollection, and IEnumerable. If the programmer needs to add something, they call for an ICollection. If they need to index, they call for an IList. If the programmer requires something that is both indexable and add-to-able, they should define their own combination of those types.
hold second i think Microsoft make this for business perspective to check in runtime if it read only collections or not (but in compile time will pass it ) it suppose to handle this runtime in derived business class
I agree, they should have added ImmutableCollection between IEnumerable and ICollection and then add an IReadOnlyList, that would have created a clean hierarchy.
But then we got the plot twist of ReadOnlyCollection not being immutable. So it should be IReadOnlyCollection. Or just skip ReadOnly for immutable classes.
@@louisfrancisco2171 just wondering, what is the point of this interface? How can it guarantee the read only feature? Only a sealed class can guarantee something like this. The type system absolutely has no role here.
@@paviad The point of it not being a sealed class is so you can implement your own IReadOnlyCollection. No interface guarantees anything about the behavior of the classes implementing it. You could write a class implementing IDisposable and have its Dispose method allocate resources instead of releasing them.
@@louisfrancisco2171 that's not how I see it, an IDisposable guarantees the existence of a dispose method, IReadOnlyCollection cannot guarantee the lack of an Add method.
@@louisfrancisco2171 I don't see it that way, an IDisposable guarantees the existence of the Dispose method, an IReadOnlyCollection cannot guarantee the lack of the Add or Clear methods (or an infinite number of other methods)
No, list can't inherit from ReadOnlyList - that would violate the history constraint. A readOnlyCollection presumably should return the same data every time it's read. A subtype that doesn't behave like that violates the LSP. You could have MutableCollection inheriting from Collection, and ReadOnlyCollection as a sibling of MutableCollection also inheriting from Collection.
I think they wanted the "common type" to be something that developers (their own developers) could use without thinking. What they should have done is IEnumerable
I remember having seen this in C# years ago, just not with ReadOnlyCollection, so this is not the only violation. This is actually sad, considering that in many regards, C# and .NET is designed very well (and I say that as someone who is not particular happy with the MS ecosystem). Even sadder, I believe I know exactly how this happened. It's a result of two attitudes. One is to not correct mistakes because corrections break existing code. The other is to get things done, also called the 80/20 principle or management making design decisions. The fact that this happened at such a fundamental level as the base data structure abstractions however is shameful.
Seems to me, and call me crazy, that if ICollection has a property that can make it a "read only" collection then IReadOnlyCollection is an unnecessary interface.
I'm going to make an IDuck interface with a Quack method and a Penguin class that implements it. Penguins can't quack so that'll be a NotSupportedException. You may THINK this is bad, but it's actually not because I didn't tell you about a "bool CanQuack()" method I put on IDuck that the gullible idiots who use my code should know that you should call first, instead of just ASSUMING that an IDuck can quack. 🙄
Ideally you'd want to separate all those functionalities into separate interfaces and have collections implement the ones that make sense. This entire hierarchy is very much a complete mess caused by maintaining backwards compatibility and some language limitations.
I don't agree. I started typing the reason why but it's too long. The only problem to me is that ICollection does not implement IReadOnlyCollection and IList does not implement IReadOnlyList
Depending on how you define the read-only interfaces that could follow LSP. But not necessarily imho 😊 I’m specifically thinking of whether you consider the interfaces to have invariants stating that the collection of items may never change. (Feel free to check out my full video on LSP for more on invariants if you have not already: th-cam.com/video/7hXi0N1oWFU/w-d-xo.html). Thank you very much for sharing your perspective 😊🙏
I support your cause. Smelly code, oily tuna design, bad Microsoft, bad! Okay, over the top but eye opening video on how to paint yourself into a corner and everyone is susceptible.
Dont know the whole interface and methods there. But I think they could have ICollection be a sub type of IReadOnlyCollection instead, then from there fork between IList and IReadOnlyList which then is implemented by ReadOnlyCollection. I believe they had a very good reason to do the way they did instead. It is hard to please everyone when developing a large framework or library.
I suppose we should please Barbara Liskov first? 😁 Thank you for sharing your thoughts. I would not be comfortable with having a mutable collection be a subtype of an immutable collection. Seems like this would lead to violating LSP by violating invariants in the supertype. But I may be mistaken. Thank you very much for watching and sharing your thoughts 😊🙏
@@ChristopherOkhravi Sure, but you could rename it as "ReadableCollection", which would make more sense in this hierarchy location, because you remove the "only" part of the ReadonlyCollection
@@ChristopherOkhravi Readonly is not the same as immutable. Immutable means it won't ever change. Readonly on the other hand only means *you* cannot change it. However I nevertheless agree that it would not make sense to derive a writable collection from a non-writable one. But that's just a matter of naming: Name the thing you cannot write to "Collection" and the thing you can write to as "WritableCollection": Then clearly a writable collection is a collection.
C++ has the “feature” of private inheritance where you can steal the implementation without promising to be a subtype. Also a lot of this mess is from mutability overload.
"The whole point of static typing is to move error from runtime to compile time because that makes the code safer". Words of a genius
Don't know if you're being sarcastic. That is precisely why static types exist and why I very strongly dislike dynamic and weakly typed languages. You have to think too much about what might go wrong. You already have to do that plenty.
@@drcl7429 Not being sarcastic, just highlighting a very good point which i have forgotten
I cannot take credit for this since it indeed is the very idea of static typing. Nevertheless I find it very important to keep reminding ourselves of that 😊😊 Thank you both for watching and sharing your thoughts 😊🙏
This is why TypeScript has appeared. An attempt to relief all the pain related to JS brining a bit of static to the primal chaos.
@@_iPilotexcept that there are too many ways to escape the static typing in typescript. It's an improvement over js, but still ultimately just papering over the gaping holes in js's type system.
I just cant you thank you enough. There is no one on TH-cam who teaches like you do. Thank you brother.
What a roller coaster ride in the explanation, especially with the entry of isReadOnly. Super awesome take on the issue and explanation!
Glad you enjoyed it! 😊 Thanks for watching 🙏😊
I *promise* you, cross my heart, swear to god, or smite me now, I violate every single SOLID principle, every single day, without fail.
😊😊😊
Dude, I just want to say thank you. Many years ago I was learning design patterns from you, and now, fast forward, I am great senior dev. My dream came true. Thank ypu for being part of my journey.
another annoying thing is that Array implements IList in C#, but when you call IList.Add it throws an exception, so you can't be sure you can safely call .Add on a method accepting an IList
So does IsReadOnly return true for Array then?
It only throws if it's full, but yes. It's annoying
You’ve earned a new subscriber.
Your explanation style is so unique and satisfying to watch, I think in mind the way you emphasize facts, and that’s why your teaching style is relatable to me.
Thank you very much. I’m glad to have you 😊🙏 and I’m glad that the content is useful 😊
What a clear explanation of complex Liskov violation on more complicated hierarchy of Collections... Hats off!.!! 😊😅
This is a brilliant explanation of LSP ! I love C#, it is a fantastic programming language, but as every programming language there are always some quirks due to historical reasons. I have heard about this "issue" before, but never explained at this level of clarity. Please keep posting these videos, preferably also using C#
Man I love you. I see that you're back on TH-cam after years. I have encountered your channel after university bullied me into Software Design, but you made it slightly more tolerable.
Anyways, thank you so much. I will be having my exam in two days. If I pass, it's totally because of you.
Even mentioned you in the Course Review section of my course, as our course was probably the most boring thing I have ever seen.
However, you made me see the beauty of it, even though I totally hated it all.
You just earned yourself a permanent member of your community :)
@chrisropher you should define the Okhravi principle: "never solve at runtime a problem that can be solved at compile time" 😊
By the way this is one of the best video I have seen on software development!!
Thank you. Those are some very kind words 😊🙏 I’m happy that the content is useful. I can’t unfortunately claim ownership of the idea since it’s very old but I will definitely steal that wording and use it. Thank you 😊🙏 and thank you for watching 😊😊
To be fair to the .NET maintainers, certain types have been added later than others and they didn't want to break backwards compatibility. For example IReadOnlyCollection and IReadOnlyList were added in .NET Framework 4.
Unfortunately, ROC has always been the weird guy.
Wow! Such a clear and logical delivery!
🙏 Thank you for watching 😊🙏
As you said, Add throwing when IsReadOnly returns true is not a violation of LSP per se. But there is a violation of LSP for ISet that also specialize ICollection.
Given an empty list of integer, when you add 1 twice then the collection has a Count of 2.
Given an empty set of integer, when you add 1 twice then the collection has a Count of 1.
Had the signature of the method been bool Add(T item), like there is bool Remove(T item), it would have been a completly different story.
Again, that's not really a violation but a very specific, arguably confusing behavior of the Add method. You can call Add and it works, that's all you need to not violate LSP.
Also, some degree of freedom is required when we are implementing an interface, not every derived class has to Add to increase the Count. In your example, Set allowing Add on existing keys is actually the better approach, otherwise your Set class would be throwing exceptions which ICollection doesn't define.
Actually, Set being an ICollection is a rather good example of trade offs in Software Engineering. By not being overly strict with how Add works, we are utilizing EVERYTHING else IEnumerable and ICollection interface gives us, and that's a lot more useful then defining a separate interface for Sets just to make it perfect.
@@ShubhamKhara I beg to differ. This is the definition of Add. It "Adds an item to the ICollection", so you expect wether the operation to fail (i.e. throw) or to succeed, meaning the Count increases. For ISet, Microsoft had to create a "new" Add method, returning true when the value is added or false when the value was already present. With this signature, the test for the contract is if Add returns true then the Count increased by 1. Regardless of the return value, calling Contains with the value must return true. This is not a trade off, this is an oversight. The trade off is that we have to live with it.
Yes, that makes sense. I had my tunnel vision set on ICollection's function's signature and not ISet's when I initially made the argument.
Thank you for the revert, appreciate it.
Man what a great explanation of the LSP. To me one of the most complex principles to understand. Thanks for the video and for the great job explaining it in a way that makes sense.
Thank you for the feedback. Much appreciated. And thank you for watching 😊🙏
wikipedia says "objects of a superclass should be replaceable with objects of its subclasses without affecting the correctness of the program"
which is pretty clear to me
It seems so complex, because there are a lot of theoretical terms and definitions to make it logically strict. But the meaning is quite simple:
Obey subtyping. A square must always be a rectangle. Not every rectangle must be a square. And then abide by that in all conditions and objects of the class or interface you are designing.
It's also basically helpful advice to developers to adhere to the subtyping contract:
- Don't forget method parameter types and return types
- Don't forget exceptions
- Don't forget input value checks and return value range restrictions
- Don't forget value restrictions on instance variables and properties - both statically, and over time from state change to state change
Of course it has applications in theoretical CS - in order to make any logical deductions, you need a strict definition. But in terms of application, it shows you what to look out for, when designing inheritance/subtype hierarchies of any kind.
But... but... what even is the purpose of implementing IList in ReadOnlyCollection? By what logic does one _want_ a "read-only collection" to necessarily also be a "list"? I don't get what the objective of that decision even was.
I don't know anything about .NET, but based on this video, it's presumably so that it can be indexed.
You can use all the other practical List methods on the ReadOnlyCollection. Plus it makes it easier to use a ReadOnlyCollection in methods or interfaces that defines the use of a IList.
For interface API design in general: If mutation is involved, interfaces should be split up into the reading part and the writing part (also works well with co/contra variance): IReadCollection and IWriteCollection. There are barely any use cases for write-only collections so that design isn't natural to most people.
Why not make ICollection just behave essentially the same as IReadOnlyCollection (and inherit directly from IEnumerable), but then have IWritableCollection inherit from IReadOnlyCollection and add just the stuff for writing to it?
Thanks for articulating this. It’s been frustrating me for years!
Collections are always a pain in the ass to implement, but I totally agree with you ! Java has pretty much the same problems with their "Collections.unmodifiableXXX" (List / Set / Map / SortedMap...) and they are also using exceptions.
Next step: how would you implement it ? I guess using composition, either containing another container inside the ReadOnlyCollection. Or by splitting collection interfaces into "permission accesses", ReadableCollection (get / iterate) WritableCollection (add / remove / clear) ? Or both ?
MutableCollection : CollectionMutator, ReadOnlyCollection
I will send this video to anyone who’d like to learn the LSP! 👏🏻
Thank you 😊🙏🙏
This video was simply fantastic, great explanation about LSP
Thank you for watching 😊🙏
Thanks for a great explanation as always. My takeaways from this are:
1. A statically typed language gives you the capabilities to design robust software. But a type checker can not do all the work for you. It hands the responsibility to the engineer to make careful design decisions.
2. It was nice to see how applying LSP leverages the power of the compiler and makes the program safer at compile time. Lots of issues could be avoided if we crafted our systems with this in mind.
Your channel is wonderful. Giving out hidden gems as insights for whoever is patient enough to hear them. Do you plan on ever continuing the design pattern series ?
I just love the way you explain things. Great vid!
Brilliant video, I love it all! This plot twist near the end made me smile. However, I feel like it's missing the explanation of Microsoft's decision about this move. As you said, there are incredible intelligent people working on .NET design, so that I am sure there gotta be a good reason for that
Thank you so much for this! I was thinking of mentioning it in your previous LSP video but I couldn't recall off hand where it was I ran into this. I even recall leaving a comment on an SO article that dug into it. I ran into this exact issue with either something from work or a side project I was working on explaining collection types. Anyways, thanks!
Very well explained. Requesting video on Liskov Substitution VS Interface segregation principle.
Before cheating off java's homework, make sure it's right
Underrated comment 😆
Java has the same, Collections.unmodifiableList or List.of
The stupid thing is Java is even worse at this. They didn't even bother adding a readonly hierarchy. They simply added factory methods that create readonly collections of type (I)Collection
Yeap, they went ass backwards, it should have been ICollection and IMutableCollection.
J-things expert, talking about c# and Dot-things.
I share your pain, though an even more fundamental violation is:
IList list = new int[] { 1, 2, 3, 4, 5 };
list.Add(27);
This also founded on dart language by google .
Theres List Type called : UnmodifiedList ... and if u call Add item on it it throws an exception same as .Net and i know it violating LSV Principal....but i also think dart is a new language why they make it like this
ReadOnlyCollection implements IList which is an interface for a mutable list. This means it throws errors if you call add() or remove().
MSFT added an “isReadOnly” flag in ICollection which allows add()/remove() to throw errors. It’s messy workaround for a LSP design error and weakens the compiler.
Chris, you are fantastic. Highly appreciate!
There's a simpler example: arrays in .NET implement IList without support for Add or Remove
That's one of the things I like in Objective-C. It's main classes are immutable: NSArray, NSDictionary; they have subclasses NSMutableArray and NSMutableDictionary.
Though I don't know .NET, I think that the hierarchy of interfaces would have been nicer if the base interface IEnumerable would have a subinterface ICollection (without add/remove), which would have two subinterfaces: IMutableCollection (with add/remove) and IList (with get(index), but without set(index, item)).
The interface IMutableList would have two superinterfaces: IMutableCollection and IList and would add the set(index, item).
No need for "unsupported operation" exception and you can immediately see what the class actually supports by its list of interfaces.
Another good reason to move to F#. No LSP issues (unless you seek them out). All functional datatypes are immutable by default (which implies readonly). It's so much simpler and more concise, once you made the paradigm shift. Rarely try/catch or exceptions and aprogram you can reason about.
Until you have infinite amount of resources and they are almost free. In the real world, all computations have their cost in CPU time, memory, and amount of data transferred over the network.
No, C# is a very good language, and it's far easier to understand than F#. This critique has a good point, but without showing us the entire design of an alternative, we can't even say it was the wrong choice.
IEnumerable, ICollection, and IList were added to .NET in version 2.0.
IReadOnlyCollection was added to .NET in version 4.5.
Is the inheritance hierarchy "wrong"? Yes. But doing it "right" would have broken the world.
If you value ideological purity more highly than backwards compatibility, perhaps C# and .NET aren't for you. 😁
Inserting an interface into an inheritance chain wouldn't break anything. I.e they could have made ICollection inherit from IReadOnlyCollection.
The big problem though, is that the class ReadOnlyCollection was introduced in .NET Framework 2.0, long before the interface IReadOnlyCollection in 4.5. So at the time they only really had IEnumerable that it could sensibly implement. They decided to implement ICollection and IList too, and those can't be removed without breaking backwards compatibility.
(I didn't bother with all the s. Too annoying to type on a phone.)
Great insights. Thank you.
Without a doubt, the ReadOnlyCollection inheriting from IList is a mistake even though the interface is technically correct by throwing that exception.
One thing I'd say in addition is, I hope no one thinks that technical validity means the correct approach to using IList.Add is to check if it's safe or otherwise handle the exception in the case of ReadOnlyCollection - we should still code with the assumption IList.Add is always safe to call, and if a client passes in a ReadOnly collection to that method, that's on them to fix.
Totally true. This happens not only with base classes/interfaces, but I have many times encountered such dead-ends in my own code when programming, and the solution is like that one (runtime and ugly). I really would like to see how to fix that.
Inheritance/interfaces are fantastic, the only big problem, for me, is the diamond problem. For example, what if you want to support passing IList to a function that expects a IReadOnlyCollection, because you want to ensure you don't to change the collection in the function that uses it. Under the current hirachy you simply can't do that, unless you go to just an IEnumerable.
How would you solve that?
Usually people end up with spaghetti code when they get things wrong. I do remember a former colleague that always wanted to use everything he knew about object oriented programming and his code was horrible. You could easily remove 50%
Speaking from memory (too lazy to test), but I think you can pass the IList straight through without any issues, and it would automatically get cast to the IReadOnlyCollection expected by the function. (IList implements IReadOnlyCollection, so the cast is valid)
Love it. I am now changing my workshop session on LSP to include this. Basically the way that C# has got round it is by a hack and use and a boolean to the ICollection contract. If you need to check the result of this first then does this go against ‘tell don’t ask’? Not a SOLID principle obviously but a commonly used rule.
Thank you for sharing this! 🙏😊
So basically, the LSP is kept, because ICollection was already broken, so ReadOnlyCollection can be just as broken. Oh the irony. It reminds me of the mathematical property that you can deduce anything from a false statement. Although this obviously has nothing in common with that, it makes sense - if you have chaos, you can deduce more chaos from it...
Coming from Swift, I'm surprised by that interface hierarchy. The equivalent protocol hierarchy in Swift is Sequence
Wrote my comment (about Objective-C), before seeing yours about Swift.
Yep, there are things, which Apple clearly did better.
P.S.: Don't like Swift (I prefer Java nowadays), but it's definitely better than mixing C and Smalltalk. :)
i don't really get the point of creating a read-only version of any type. If you want to make an instance of a type read-only you could use the const keyword and mark the methods which don't change the state of the object also with const. Or is the const keyword not available in C# (I'm coming from C++ world)?
c++ has a great way of dealing with readonly. they thought really well the const keyword for methods
In C#, you can mark the methods of a struct with the readonly keyword.
My perspective is that it helps to reduce unintended side effects.
Say I created function ProcessData(IList list). Since the list is passed by reference, there's nothing stopping me from modifying the list in ways that someone calling the function might not expect.
By contrast, if I have a ProcessData(IReadOnlyList list), then that communicates a guarantee (ignoring reflection / recasts) to the caller that the list will not be modified.
Side effects can be VERY tricky to debug and hunt down, so imho, it's generally a good idea to reduce those opportunities as much as possible.
Your explanation was engaging and thought-provoking.
Made me wonder about the relationship between number of layers of abstraction and probability of violating the Liskov Substitution Principle. For "real-world" solutions of sufficiently complex problems, is this curve likely exponential, linear, or ...?
Hi chris nice video, I have a question that might be a bit off with the topic of this video, the thing is that i started to read a book called "Dependency Injection: Principles, Practices and Patterns" in wich the author explains the different techniques we can use in order to inject properties into our clases (Constructor injection, Method injection and Property injection), he also explained some anti patterns related to dependency injection (Control freak, service locator, ambient context and constrained constructor), all off these stuff its explained on the chapter 4, 5 and 6 and i think you could create a great video explaining thoso topics
@ChristopherOkhravi Can you please explain what could be the possible reason of Implementing IList by ReadOnlyCollection base class.
I didn't get this part
IReadOnly** was added much later - after they had made that big blunder with `IsReadOnly`. It's also because (and this might change in .Net 10) IList and ICollection does not implement their IReadOnly counterpart.
this could be an interesting case for type-families (ts ternary type, or rust traits). In fact, typescript has a utility type that does exactly that. i don't think it will be very long until c# gets functions at the type level.
At first glance I'd like to turn the IEnumerable, ICollection and IList upside down from what you did draw. Mostly because I see the IEnumerable as the root and that it then grows upward. Arrow directions can also be confusing - is the arrows the way the functionality evolves or the way the drill-down occurs. It comes down to the way you think, and it's easy to get locked into one pattern (information flow for example) and if someone else uses a different pattern then it takes time to adapt. At least you did set a baseline in the first 2 minutes.
As for the inheritance issue I'd like to change so that the indexing that IList provides should be provided by IEnumerable and the IReadOnlyCollection should be between IEnumerable and ICollection.
.NET concrete collections have always been a mess. Collection implements List, SortedList implements IDictionary and those are the ones just off the top of my head.
In Swift, "MutableCollection" inherits from "Collection". Seems like .Net could solve some conceptual problems if they did the same. Well, "IMutableCollection", I suppose, since MS seems to think the name defines what something is as opposed to just looking at the AST and checking if it's an interface or a class. Remindsof them thinking i can change a file from Word to a PDF just by renaming it.
I also would rename IEnumerable to IIterable, coz it sounds like we have to deal with something "numbered", but we actually can only move next...
Why isReadOnly means no LSP violation?
I saw one example of LSP violation myself in .Net source code (when of the methods of base class was not appropriate for the subclass and it was just throwing exception)
It's like the first thing that gets violated in OOP and probably the least important TBH
I think you were right first time. I don't know why they made its return value to be void, could have just returned 0 or maybe -1 on add() then. At least Java uses a boolean.
Very nice video as always, it definitely was food for thoughts.
Letting alone the discomfort I would have using an object of a class that could be (in theory) both mutable and immutable and letting alone the cognitive overhead in terms of counterintuitive naming, I could not help but thinking that this way of deciding the behavior of an object of a given class at runtime with an if-else block may be a little bit of an anti-pattern in a language that allows you to use types (at compile time) to address the issue. What I am wondering now is which reason(s) they may have had to make this particular design choice. Any idea?
What a great video!!
Doesn't it break ISP?
Read only collection is forced to implement add and write method.
Christopher, awesome presentation, with the gestures you remind me a little bit of Morshu :) I want to ask a question because even chat bots confure this one. Is there a circular reference between string and object's ToString? String inherits from object, but object cannot be compiled without string. What do you think?
Regarding the issue in the video, I think that F# solved this issue and applied a much more pragmatic definition of interfaces. There wasn't a need for interfaces, rather the type must have a method that matches the one that is called in order to compile, without the need of interfaces, making implementations very specific and at the same time, very generic, but I might be wrong, haven't checked.
Amazing video as always!
Hey, just wanted to let you know that the link on recommended books, placed on your website, is not working
I mean why/when would you require a read-only collection anyway? If you're going to only read through a collection, you don't really care if it's read-only or not, and limiting it to only read-only collections through the type-system would do more harm than good.
I think it's just a hard problem to solve, if not impossible, would love to hear a better solution though if anyone has it. Just as a side note, Java solves this in a similar way: UnmodifiableCollection implements Collection.
One common use case would be to communicate and enforce a contract: my library function takes a collection, does some computation using its elements and returns some result to you. In the contract of this function, I promise not to alter any collection you pass me. This is important, because you can now pass me a collection without having to worry (and verify) whether it has changed after I'm done with it.
If we have interfaces defining read-only collections, I can both document and enforce this contract succinctly and reliably in one place. Conversely, as it currently stands in C# and Java, you basically have to both read the (prose) documentation of my function to find out this part of the contract and trust my library to actually adhere to it. This makes the contract at the same time weaker _and_ harder to discover, neither of which is a desirable quality in this context.
In short, I may not care much if a collection you pass me also supports modification if I only need to read it, but you should care whether I can modify a collection you pass me or not.
I'm curious what you imagine the harm done to be of defining a ReadableCollection interface that is extended by a MutableCollection interface which simply adds the add and remove methods. If my function must somehow mutate your collection in order to fulfill its goals, it can define a parameter of type MutableCollection, which also implicitly communicates to you that I intend to alter your collection. If you already have a MutableCollection but my function only requires a ReadableCollection (and defines its parameter accordingly), you can simply pass me your mutable collection with the reassurance that I still can't modify it, without any additional effort on your part.
As to the hard problem to solve, maybe you can elaborate on what exactly that problem is?
Thats how the entirety of functional programming works and increasingly considered the norm. Don't mutate your arguments.
@@silberwolfSR71 Don't you see the problem with what you're saying? "...defining a ReadableCollection interface that is extended by a MutableCollection interface". If so, the LSP is "violated" once again, and in a WAY worse way, IMO.
@@adambickford8720 Okay but C# isn't a functional language? The LSP doesn't refer to functional programming either. I don't really see your point.
EDIT: just to clarify, I see the purpose of immutable collections, of course. But in a hierarchy structure like it's defined in C# (IEnumerable, ICollection, IList, etc), I don't see the point of having an LSP-compliant read-only collection.
@@numeritos1799 What do you think linq is?
Get educated before coping an attitude, mmkay?
So could the developer toggle ICollection.readonly to true or false dynamically so at times the collection is readonly? That would explain the exception at runtime.
IsReadOnly is usually a read-only property.
ReadOnlyCollection implementation always returns true.
List implementation always returns false.
A Collection, when instantiated with a list parameter, just returns the IsReadOnly value of that list.
But you could implement it as read-write in your own class if you need to.
Also, arrays are collections that cannot be added to, but can be changed by index. Readonly collections cannot be changed by index. The implementations of ICollection and IList are basically flipped. It would have been better concentrate the mutability methods and properties in the most top level type.
IReadOnlyCollections stems from this historic mistake, with ReadOnlyCollection being a simple wrapper around every other collection type to make it work. You wouldn't use it in any declaration.
If ICollection has a read_only property, then what's the point of IReadOnlyCollection? And yes, it's a violation.
I expected you show at the end of the video how your solution would be.
Thank you I really enjoyed the video, especially the inreractive parts!
I guess how I would try to solve it: we would want to introduce an iindexer interface and let IList implement that, then also let the ReadyOnlyCollection implement that interface to "solve" this? (With the information that is used in the video)
Any plans to continue the object oriented lectures?
My sincere apologies but the answer is unfortunately no. I recorded everything but it didn't turn out as good as I wanted it to. I've got a few things in the works however so it is possible that I might redo it from scratch but that's far out into the future. Again, my sincere apologies.
@@ChristopherOkhravi no need to apologize, your content is awesome
I think there are 2 ways of thinking about read-only collections. On the one hand, these are generalizations of read-write collections, which only support reading. On the other hand, these are collections which guarantee they won't change.
I believe MS devs have change their mind about it couple of times and here's the result
This could be resolved easily with Haskell's static typing. Pretty trivially, actually, so that violators fail at compile time.
an isReadOnly() function is a bad idea. The read-only quality should be embodied by the typing directly.
i really like this video about these horrible decisions and the wise consideration to use types.
You didn't suggest a fix.
Inheritance among interfaces appears to be bad design. When I have something that implements IList, all I really care about is the extra stuff it adds on top of its parent class, so it shouldn't have a parent class. Something that today implements IList, should tomorrow implement IList, ICollection, and IEnumerable. If the programmer needs to add something, they call for an ICollection. If they need to index, they call for an IList.
If the programmer requires something that is both indexable and add-to-able, they should define their own combination of those types.
what is his name?
Great video keep them comming
hold second i think Microsoft make this for business perspective to check in runtime if it read only collections or not (but in compile time will pass it ) it suppose to handle this runtime in derived business class
I agree, they should have added ImmutableCollection between IEnumerable and ICollection and then add an IReadOnlyList, that would have created a clean hierarchy.
But then we got the plot twist of ReadOnlyCollection not being immutable. So it should be IReadOnlyCollection. Or just skip ReadOnly for immutable classes.
That atrocious design still can't beat Windows 11 start menu
Googling windows 11 start menu since I have not been in Windows for years 😊😊
Did I miss something or someone actually made read only collection and then implemented add / remove methods with exceptions?
Yes. System.Collections.ObjectModel.ReadOnlyCollection implements IList and all modifying methods throw unconditionally.
@@vytah I think I'll stick with JavaScript for the time being.
Hello my friend and thank you for your new video. It’s very interesting. Would love to hear something interesting about IQueryable.
Cool as usual. I wonder how one interface can inherit from another 😊
The signatures are inherited. The subtype interface contains all signatures of the parent. Thanks for watching 😊😊
Maybe if we had conditional types like typescript in C#, programmers would be able to deal with it more effectively
Why is ReadOnlyCollection implementing IReadOnlyCollection and not simply IEnumerable? Who else implements IReadOnlyCollection?
Apparently, over 70 classes implement it.
@@louisfrancisco2171 just wondering, what is the point of this interface? How can it guarantee the read only feature? Only a sealed class can guarantee something like this. The type system absolutely has no role here.
@@paviad The point of it not being a sealed class is so you can implement your own IReadOnlyCollection. No interface guarantees anything about the behavior of the classes implementing it. You could write a class implementing IDisposable and have its Dispose method allocate resources instead of releasing them.
@@louisfrancisco2171 that's not how I see it, an IDisposable guarantees the existence of a dispose method, IReadOnlyCollection cannot guarantee the lack of an Add method.
@@louisfrancisco2171 I don't see it that way, an IDisposable guarantees the existence of the Dispose method, an IReadOnlyCollection cannot guarantee the lack of the Add or Clear methods (or an infinite number of other methods)
Mmm... so... the right way of doing that would have been IEnumerable
No, list can't inherit from ReadOnlyList - that would violate the history constraint. A readOnlyCollection presumably should return the same data every time it's read. A subtype that doesn't behave like that violates the LSP. You could have MutableCollection inheriting from Collection, and ReadOnlyCollection as a sibling of MutableCollection also inheriting from Collection.
When I hear LSP I think language server protocol
[13:16] "If we have good Type hierarchy"....🤥
I think they wanted the "common type" to be something that developers (their own developers) could use without thinking. What they should have done is IEnumerable
What an awful solution - simply don't inherit in interfaces and the problem evaporates. Your idea leads to combinatorial explosion.
@@Asdayasman I don't think it does.
Great Video
the whole problem starts and ends with inheritance, this time is not inheritance between classes but between interfaces
The issue is not inheritance but subtyping. LSP applies as soon as there is subtyping. Thank you for watching and for sharing your thoughts 😊🙏
I remember having seen this in C# years ago, just not with ReadOnlyCollection, so this is not the only violation.
This is actually sad, considering that in many regards, C# and .NET is designed very well (and I say that as someone who is not particular happy with the MS ecosystem).
Even sadder, I believe I know exactly how this happened. It's a result of two attitudes. One is to not correct mistakes because corrections break existing code. The other is to get things done, also called the 80/20 principle or management making design decisions.
The fact that this happened at such a fundamental level as the base data structure abstractions however is shameful.
Excellent. Sometimes I feel so stupid making this kind of mistakes. Now I realize, I shouldn't be too hard on me :)
I feel the same way 😊
Seems to me, and call me crazy, that if ICollection has a property that can make it a "read only" collection then IReadOnlyCollection is an unnecessary interface.
OOP is upside-down. Types are generalizations of objects, objects aren’t realizations of types.
Great video!
I'm going to make an IDuck interface with a Quack method and a Penguin class that implements it. Penguins can't quack so that'll be a NotSupportedException. You may THINK this is bad, but it's actually not because I didn't tell you about a "bool CanQuack()" method I put on IDuck that the gullible idiots who use my code should know that you should call first, instead of just ASSUMING that an IDuck can quack. 🙄
Nailed it!
Ideally you'd want to separate all those functionalities into separate interfaces and have collections implement the ones that make sense. This entire hierarchy is very much a complete mess caused by maintaining backwards compatibility and some language limitations.
I don't agree.
I started typing the reason why but it's too long.
The only problem to me is that ICollection does not implement IReadOnlyCollection and IList does not implement IReadOnlyList
Depending on how you define the read-only interfaces that could follow LSP. But not necessarily imho 😊 I’m specifically thinking of whether you consider the interfaces to have invariants stating that the collection of items may never change. (Feel free to check out my full video on LSP for more on invariants if you have not already: th-cam.com/video/7hXi0N1oWFU/w-d-xo.html). Thank you very much for sharing your perspective 😊🙏
I support your cause. Smelly code, oily tuna design, bad Microsoft, bad! Okay, over the top but eye opening video on how to paint yourself into a corner and everyone is susceptible.
Dont know the whole interface and methods there. But I think they could have ICollection be a sub type of IReadOnlyCollection instead, then from there fork between IList and IReadOnlyList which then is implemented by ReadOnlyCollection.
I believe they had a very good reason to do the way they did instead. It is hard to please everyone when developing a large framework or library.
I suppose we should please Barbara Liskov first? 😁
Thank you for sharing your thoughts. I would not be comfortable with having a mutable collection be a subtype of an immutable collection. Seems like this would lead to violating LSP by violating invariants in the supertype. But I may be mistaken.
Thank you very much for watching and sharing your thoughts 😊🙏
@@ChristopherOkhravi nah, Barbara Liskov can go cry about it
jokes aside, it probably was made this way to not break existing code
@@ChristopherOkhravi Sure, but you could rename it as "ReadableCollection", which would make more sense in this hierarchy location, because you remove the "only" part of the ReadonlyCollection
@@Scorbutics You'd probably to rename it to just Collection then. Presumably all collections are readable.
@@ChristopherOkhravi Readonly is not the same as immutable. Immutable means it won't ever change. Readonly on the other hand only means *you* cannot change it.
However I nevertheless agree that it would not make sense to derive a writable collection from a non-writable one. But that's just a matter of naming: Name the thing you cannot write to "Collection" and the thing you can write to as "WritableCollection": Then clearly a writable collection is a collection.
C++ has the “feature” of private inheritance where you can steal the implementation without promising to be a subtype. Also a lot of this mess is from mutability overload.