Yeah, now that you put it that way, that's true too. I guess I was approaching it more from the enjoyment angle; I definitely enjoy the prototyping/sketching out ideas more, but maybe I just have a personal problem with finishing things to completion...Opposite experience there. "Anybody" can do the bootstrapping of most projects.
Making it polished, correct, maintainable yet optimized and shiny is where experienced engineers shine the most.
Well, CS is just a collection of techniques people have invented to make programming a little less unmanageable. It's a pretty infantile science. Most of it is cargo cult faggotry (like Lisp) dressed up as "science" (pretty much like climate science, modern physics, epidemiology, political science, economics, psychiatry...)I don't think typing is the same as what I'd like to do, but I'm not overly familiar with the specific meaning of CS terms
We all have, that's why we're getting paid :DYeah, now that you put it that way, that's true too. I guess I was approaching it more from the enjoyment angle; I definitely enjoy the prototyping/sketching out ideas more, but maybe I just have a personal problem with finishing things to completion...Opposite experience there. "Anybody" can do the bootstrapping of most projects.
Making it polished, correct, maintainable yet optimized and shiny is where experienced engineers shine the most.
This doesn't really have to do with interfaces so much. Interfaces are about creating highly generic code for frameworks and such, so third party users can implement their own classes while still using your framework. Code readability, on the other hand, is much more affected by the quality of your code:
1. Writing clean, logical code.
2. Adding relevant comments in key places.
3. Splitting up code/data into manageable pieces (whether classes, methods, etc). OOP is great for this, btw.
4. Using descriptive names for variables/methods.
5. Avoiding "cute" shit and keeping code straightforward, e.g. writing very complex code statements using some of the language's tricks, or things like lambdas might seem leet in the moment, but it really fucks with the code's readability.
If you do the stuff above, your code should be fairly easy to understand.
Wrong type of interfaceInterfaces are really imporant in HW industry. USB-C port is a standard interface, which allows any standard connector to be plugged into it. (When the connector isn't manufactured by China by XDUO company, then you need a different cable with more connector like connector.)
This doesn't really have to do with interfaces so much. Interfaces are about creating highly generic code for frameworks and such, so third party users can implement their own classes while still using your framework. Code readability, on the other hand, is much more affected by the quality of your code:
1. Writing clean, logical code.
2. Adding relevant comments in key places.
3. Splitting up code/data into manageable pieces (whether classes, methods, etc). OOP is great for this, btw.
4. Using descriptive names for variables/methods.
5. Avoiding "cute" shit and keeping code straightforward, e.g. writing very complex code statements using some of the language's tricks, or things like lambdas might seem leet in the moment, but it really fucks with the code's readability.
If you do the stuff above, your code should be fairly easy to understand.
You're on the money about interfaces being more for communicating with API's etc. If you're not making a library and your codebase is full of interfaces, you have a problem.
One piece of advice I like to give out to novice programmers is:
INTERFACES DO NOT EXIST
I mean yeah C#, Java and a few other languages implement "interfaces", but they are just less useful base classes.
C++ doesn't have (or need) interfaces because it has multiple inheritence.
There is no special concept of "interfaces". They aren't things you use with a specific name and a specific purpose. There is no such thing as a "contract" within programming (or at least, not as a distinct feature, just a renaming of polymorphism). Interfaces exist to allow polymorphism, that's it. Classes already do polymorphism and they do it much better, since in many cases implementing an interface means re-writing a similar-but-slightly-different version of a function in every class that implements it, rather than having proper functionality in a base class (or, even better, a delegate object, which is usually a much better fit for the job than inheritance). I have literally seen cases where people have rewritten the exact same function verbatim in multiple classes that extend the same interface. Thinking about everything as a contract hugely encourages developers to try and define everything as an interface for every little concept in their codebase, and they end up with huge amounts of interface bloat.
Most people see interfaces as "the" correct way to handle conformity to a certain design, even though (in my opinion) abstract base classes work just as well if not better, since they can easily handle default cases (without needing things like the null object pattern).
I largely agree with the idea behind interfaces - you should always split up your functionality into separated functions, classes and modules, and in order to make inter-class communication easier you need polymorphism, which interfaces utilise. But this is true with or without interfaces. Whether I inherit from a base class or implement an interface, logically it makes no difference because there is no fundamental difference. In both cases I can make a function take the base type and pass in any child of that type, whether it inherits from it or "implements" it.
Many books have been written on the correct time to use "interfaces" vs base classes. Whole stack overflow threads have arisen around the "logic" or "philosophy" of interfaces vs classes, and it's entirely bullshit. There is no difference. You should always use a base class if you have that option because you never know if you will need default functionality later. I find this crops up a LOT more often than the occasional time where you need to inherit from a base class but are already using the one available to languages like C# or Java. Now that C# has implemented default behaviour in interfaces, they are literally identical to base classes, which is stupid and shows just how much of a meme interfaces are.
The biggest lie in the software industry is the lie created by Java that there are these magical things called "interfaces" which are somehow different to good old fashioned polymorphism. Actually the biggest lie is that Python is a usable language, but it's one of the bigger lies.
Will most abstract base classes in C++ have next to no functionality? You bet. And they will look exactly like your typical C# or Java interface. Because interfaces are just that concept, given a name. The difference is, my abstract "vehicle" class (which Car and Truck both extend) makes real-world sense and is a simple structure. Thinking about it as contracts will inevitably lead some developers to instead make IDriveable, ISteerable, IPowerWindowsController etc interfaces, because they are thinking about how their vehicles will interact with their program, rather than as actual objects with a purpose. Even better, I can actually add default functionality to these classes and extend "backwards" as well as forwards, because there is absolutely functionality that all vehicles will need, and if it's added to the base class, they ALL get it automatically. Furthermore, I can make my vehicle class inherit from other base classes and gain even more default functionality (such as a "Movable" class with basic functions for setting positions and velocities, etc), something I cannot do with interfaces (which can only inherit off each other and can add no functionality, only contract). Interface thinking encourages programmers to make a whole bunch of really small pseudo-classes in a flat structure, which seems like a really good idea at the time until you realise that the entire point of inheritance is to inherit and overwrite functionality. "contract" thinking instead encourages you to slap on additional responsibilities to existing classes by making them inherit another interface, so they will work with some function that expects an interface of that type. This encourages god classes and overall messy design.
Please stop talking about interfaces like they are an actual thing. They are not. People who frame everything in terms of "communicating interfaces" and "contracts" and "roles" rather than a simple (mostly a simple tree) structure of objects frequently end up with lots of extra interface bloat, and don't benefit much from it. Interfaces only exist because language designers are too lazy/incompetent to solve the multiple inheritance problem (despite C++ solving it decades ago). If your language forces you to use interfaces in certain cases, by all means use them, but the only time you should be actively creating interfaces is when you NEED a second base class (which should be rare since you should be preferencing composition over inheritance anyway), and then you should use the "extract interface" feature of your editor. In all other cases they are pointless.
In most cases, your best bet is to create simple classes with 1 responsibility each, and the minimum amount of inheritance possible since inheritance is an inherently dangerous operation - it's the strongest coupling a class can have, after all - and avoiding it at all costs is a worthwhile endeavour. Instead, make classes themselves simple, delegate other responsibilities to other classes, and pass in any you need through the constructor. No interfaces necessary. Interfaces should be used rarely, and when necessary, don't center your design around them (or complex inheritance systems in general).
I guess I don't really hate interfaces as a language feature, I understand they exist to overcome a limitation in the languages design, I mainly hate the way they have become this sort of "new idea" and "design approach" around making everything a contract and defining all your objects as a set of functionalities rather than as discrete objects with specific responsibilities. Multiple Inheritance, while possible in C++, is rarely used because there's just not that many uses for it. Meanwhile in C# land I see classes frequently sporting 4 or 5 interfaces because the developers decided that some class (usually a god class like "Player") has to fulfill a bunch of different contracts rather than splitting up their design properly into discrete parts.
"Interface" as a concept IS a real thing and is important. If I have written an API or library for other developers to use, and I need to make a change, I need to carefully consider the interface I am presenting, because a change to a function signature or class name can break projects. This is not to be confused with "Interfaces" as a language feature.
ALL THAT SAID.
I mostly agree with all your points. You should split your design up into neat, usable interfa-*cough*-objects, each as simple as possible, and pass them around constantly using some sort of messaging or communication system and polymorphism. You don't need fancy language tools, you don't need "design patterns" like service locator (which is actually an anti-pattern, don't use it) or singletons (anyone who uses a singleton for any reason needs to be fired immediately and barred from the software industry). The biggest mistake most programmers make is sticking to "known good designs" rather than just using the simplest thing that works, and it ends up bloating their code and making it impossible to use or extend in any other sort of paradigm. What should be as simple as passing one class to another through a constructor often becomes multiple factory classes, a fancy dependency injection system, and strange looking classes like CarClassGeneratorGenerator<T>, which you don't ever want. This happens because someone read somewhere that factories and DI are good tools (which they are, in moderation), and it turned their tiny usable system into an unmaintainable mess. Don't get me wrong, studying design patterns can be extremely useful when looking for techniques for solving certain complex problems easily. Factories make a lot of sense when you need to configure objects in complicated ways. Service-based systems make sense when you have to conform to corporate APIs or a rigid centralised database structure. But when people see their favourite design pattern as -the- correct way to program, problems arise. "Contract based" programming against interfaces is one such design pattern. Don't fall into it's trap. It's excellent in some cases - for example in Unity having an IShootable interface that components can implement so that a raycast can call an OnGetShot message on every IShootable component it hits is a great design. But don't buy into the meme of putting interfaces everywhere, a lot of the time they will just bloat your code. The C# standard library suffers from this bloat. What is the fundamental difference between an IEnumerable<> and a IReadOnlyList<>? I know the answer, you don't have to tell me, as they implement different functionality but it's a very complex heirarchy setup which is difficult to learn when starting the language, and in most everyday cases people will just cast to an IList<> and be done with it.
Also keep in mind: Comments should largely not be used, ever. In the best cases they explain an algorithm that should already be understandable. Usually people use a comment above a function saying "//this adds 2 numbers together" entirely because their function is called NumberCrunch(int input1, int input2) rather than Sum(int a, int b). In almost every case I see in my day job where someone has used comments to explain code, they would have better spent their time making their code more readable rather than using variable names like i and then having to explain what i is used for.
They are genuinely useful in cases where code must (by the very nature of the problem) be complex. Unless you're writing something like a fast inverse square root on a daily basis, these cases are extremely rare (ironically the Q_rsqrt function could have done with more actual comments and less "what the fuck" comments). The far more common really good use case for comments is metadata. If you have to do something in a certain way because of some other system, mention it (such as having to do something strange in order to work around a bug in a third-party API that someone reading just the code won't know about). If you need to conform to some third-party API, by all means add a link to it's documentation in a comment. Comments should be there to augment your code with extra useful information that is relevant to, but not contained within, the code. They are not there to explain the code, as the code is already designed to be a human-readable format for understanding the logic of the problem (the common misconception is that code is written in a way which is designed to be understandable to computers. That is false. Programming code is designed to be easy and efficient to understand for humans. We use a compiler to make it readable by computers). Using a comment to explain hard to understand, badly written code is like writing a book to explain what a previous, difficult to understand book actually meant. Just rewrite the first book!
Good god man, what a wall of verbal diarrhea! Just kidding, I kinda agree with much of it. I've always hated design patterns for example, not because they are always a bad idea (sometimes they are a great solution for some problems), but because they tend to put programmers (especially novice ones) into this mindset of "I gotta use all of these design patterns for every little shit", and yeah, that leads to horrific ugly-as-fuck needlessly obfuscated code. If you also think about writing code like writing literature (yes, I know they are different things), design patterns would be the loose equivalent of filling your prose with other people's paragraphs, which takes all the fun out of it. Design your own shit (unless you are stuck), it's a lot more fun. The great thing about doing it this way is as you gain more experience, you will independently arrive at some of the design patterns, and that will feel a lot better than just mindlessly trying to insert them into your code everywhere.
He goes on further to explain what he means by Java Culture:Nevertheless, we don't plan on adding a macro facility to Java any time soon. A major objection is that we do not want to encourage the development of a wide variety of user-defined macros as part of the Java culture.
The advantages of Java is that it easily serves as a lingua franca - everyone can read a Java program and understand what is going on. User defined macros destroy that property. Every installation or project can (and will) define its own set of macros, that make their programs unreadable for everyone else. Programming languages are cultural artifacts, and their success (i.e., widespread adoption) is critically dependent on cultural factors as well as technical ones.
We are catering to the Java culture, while trying to manage things well on the technical side at the same time. In general, once can contrast the Scheme-like philosophy of using a small number of very general constructs, with the more mainstream approach of having a great many highly specialized constructs, as in C or Modula style languages.
Java is clearly in the latter camp. Most Java developers are happy to have dedicated, narrowly focused solutions that are tailored to a specific problem. I am keenly aware of the drawbacks of such an approach, but I don't see it changing very quickly.
Java isn't the most pleasant thing.. but I have a suspicion you probably wouldn't want to work with C++ too much either if that C++ was written by the same people who use Java today. Unless you're a lone wolf maintaining a small program/module, you gotta deal with other people's mess, and when that mess looks like a four level of indirections pointer I'm getting outta here.I hate working with Java so much.
That's all I wanted to write...
At least it pays well.
I wouldn't categorize Java and C# as the same thing. The latter is obviously superior at this point. I do agree with the point about bloat and I suspect there are language developers at Microsoft who get paid bonuses based on how many new things they manage to add to the language. Too many basic operations can now be done multiple ways which is bad for readability and backwards compatibility in future.SomeGuy:
Sometimes, less is more. Yes, you can do what interfaces do with C++'s inheritance systems. But languages that restrict inheritance (particularly, inheritance of implementations, which can lead to the diamond problem) do it for a reason. That same reason also explains whenever a language doesn't have operator overloading (which can lead to unreadable code in the hands of someone who uses this stuff outside of obvious math like operations) and other features that can quickly become.. misfeatures in the hands of incompetent programmers working in large teams where 50+ people need to be able to read whatever shit the colleague wrote.
Your rant has some good points.. when interpreted outside of the context of programming language design. Because within the context... the language designers understand the issues you raise and they think the language is better off without the features for the audience they target.
Java was designed to be usable in the hands of morons.
To quote one of Java's language designers, Gilad Bracha, on why they wouldn't implement anything that could resemble a macro system :
He goes on further to explain what he means by Java Culture:Nevertheless, we don't plan on adding a macro facility to Java any time soon. A major objection is that we do not want to encourage the development of a wide variety of user-defined macros as part of the Java culture.
The advantages of Java is that it easily serves as a lingua franca - everyone can read a Java program and understand what is going on. User defined macros destroy that property. Every installation or project can (and will) define its own set of macros, that make their programs unreadable for everyone else. Programming languages are cultural artifacts, and their success (i.e., widespread adoption) is critically dependent on cultural factors as well as technical ones.
We are catering to the Java culture, while trying to manage things well on the technical side at the same time. In general, once can contrast the Scheme-like philosophy of using a small number of very general constructs, with the more mainstream approach of having a great many highly specialized constructs, as in C or Modula style languages.
Java is clearly in the latter camp. Most Java developers are happy to have dedicated, narrowly focused solutions that are tailored to a specific problem. I am keenly aware of the drawbacks of such an approach, but I don't see it changing very quickly.
Mind you, people who work on such language designs aren't moron themselves (Go, which is basically the new java, was created by Ken Thompson and Rob Pike), but they unabashedly cather to morons.
Any argument made that X feature is just a weaker version of Y, or how you could do a better, more powerful abstraction is beside the point in the views of those who design those languages : they are intently trying to eschew as much as humanely possible any feature that can lead to code that takes even 1% more effort to read. Languages like Java are verbose but one can't deny that you more or less know exactly what a piece of code in front of your eyes does without having to go 10 layers deep into the other functions it calls, objects it constructs and so on.
The same can't be said of code written by people who have fun with C macros and pointer arithmetics, or C++ template meta programming.
As for C++ itself.. it's a good language that has its uses, it's easier to write high performance programs in it (because few of its abstractions have noticeable costs), but it has grown into a massive behemoth, most C++ developers don't really know all the features of the language spec and large companies have as large styleguides that prohibit the use of like a third of the language spec.
Google's style guides for C++ for example prohibit the use of exceptions, RTTI, multiple inheritance and severly discourage raw pointers, macros, type inference (avoid obvious repetitions, but don't replace every type declaration with auto), operator overloading outside of very specific uses. Most corporate styleguides for C++ focus more on features that aren't allowed, or features you should think twice and thrice before using (like template metaprogramming, the google styleguide even has a "boost libraries leads to unreadable code, but some of the libraries are useful, here they are listed, do not use the libraries unmentioned:").
Of all the PL in the world, I think C++ is the only one that has so many styleguides out there dedicated to the the ban on the use of its features.
'nyhow, java and c# are old news, in their attempt to curtail verbosity they have grown too much in features for the morons they were designed for, hence the language Go, which is even simpler conceptually and more straightjacketed than the first release of Java. Life is a repeating cycle, Go is finally introducing generics after doing without and saying they shouldn't exist for a whole decade since its creation, 20 years from now Go will ressemble C# and another language will arise to target the corporate drone.
The linux kernel will probably still be written in C and most useful desktop software like browsers, office suites, image editors, video editors will still be written in C++.
Same.I hate working with Java so much.
That's all I wanted to write...
At least it pays well.
Java (and C# and other languages like that) are meant for large corporate teams, with varying levels of developer quality and a lot of dull, routine work. It's not a fun language to work with, because the base language is purely functional rather than elegant, and has a lot of stuff that takes way too many steps to do (though they have been improving this over the years). And then on top of that, you have all the frameworks that a typical modern Java project must use (e.g. Spring, Hibernate, EJB, etc), and they add a ton of boring overhead to an already boring base language. The average corporate Java or C# programmer probably spends like 80% of their time on boilerplate code, which severely limits their ability to try anything new, experiment, play around. Oh you want to try this approach instead? Well good fucking luck refactoring all that plumbing code, the configuration files, the annotations, etc.Java was designed to be usable in the hands of morons.
Java is neither functional nor elegant....It's not a fun language to work with, because the base language is purely functional rather than elegant...Java was designed to be usable in the hands of morons.
Hahaha at my current company they use Lombok which adds fucking getters and setters to everything. That you can't see because they're added during compile time...Same.I hate working with Java so much.
That's all I wanted to write...
At least it pays well.
Literally every feature of Java is broken in some way. The only awesome thing about Java is probably their enum classes. I'm very lucky that I was able to get out of my Java job and into a C# Unity gamedev job. C# is basically just Java but better anyway.
It's why I chuckled at the guy above who was like "they put in interfaces for a reason". Yes. That reason was incompetence.
Also, controversial opinion time (or, even more controversial opinion time): "Add Getters/Setters" features in editors are a mistake and you shouldn't use them as they encourage you to bloat your classes with needless functions that undermine your abstraction and essentially make your classes into data objects with direct access to their fields. In general you (for the most part) shouldn't be using getters and setters (or any amalgomation of them such as properties). Getters (and properties) I can understand, and I use them occasionally for getting data out of a class sometimes, but setters are almost always entirely useless. The interface to your class should be driven by it's functionality, not it's data. So you should tell a library to OrganizeBooks, not "SetOrganised" to true. Setters usually provide no opportunity for expansion, are horrible for encapsulation, and once you add a getter and a setter for a class member it may as well be a public variable, since it's the same thing.
The reason this is relevant is because during my Java job it seemed that every single class I worked with was basically a bag of data with a very thin veneer of functionality on top of it. 90% of the functions in any given class were for setting or reading a piece of data - effectively just getters and setters. All the real work was done in monumental and horrendously disorganised service classes. It was a nightmare to work with. The service model generally sucks if not done perfectly, but this was even worse than usual. They used Spring though, so the code was already going to be bad (Spring is based on Singletons - which makes it horrible by default because Singletons are nothing but an anti-pattern).
I don't know if that's a problem with Java coders in general, or if this place was just particularly shitty. I have very little respect for Java developers normally because I find that the "Java community" mostly embraces awful frameworks and terrible coding styles, and nobody ever stops to think "hey why does this framework suck so much?"
Compile time, or runtime?Hahaha at my current company they use Lombok which adds fucking getters and setters to everything. That you can't see because they're added during compile time...
Could be linked with how IT is taught. If you don't have quality lessons on how to leverage OOP strengths then the intuitive way is to cut short through the mandatory class boilerplating and add getters and setters everywhere so you can resume writing in pseudo C.Also, controversial opinion time (or, even more controversial opinion time): "Add Getters/Setters" features in editors are a mistake and you shouldn't use them as they encourage you to bloat your classes with needless functions that undermine your abstraction and essentially make your classes into data objects with direct access to their fields. In general you (for the most part) shouldn't be using getters and setters (or any amalgomation of them such as properties). Getters (and properties) I can understand, and I use them occasionally for getting data out of a class sometimes, but setters are almost always entirely useless. The interface to your class should be driven by it's functionality, not it's data. So you should tell a library to OrganizeBooks, not "SetOrganised" to true. Setters usually provide no opportunity for expansion, are horrible for encapsulation, and once you add a getter and a setter for a class member it may as well be a public variable, since it's the same thing.
I like your rants and I generally agree, but I do have to call out that in game development there are times when singletons make sense. For example, if it's a preferred pattern in the game engine you are using to have global access to something like a Settings or Game State object that is auto-loaded.Good god man, what a wall of verbal diarrhea! Just kidding, I kinda agree with much of it. I've always hated design patterns for example, not because they are always a bad idea (sometimes they are a great solution for some problems), but because they tend to put programmers (especially novice ones) into this mindset of "I gotta use all of these design patterns for every little shit", and yeah, that leads to horrific ugly-as-fuck needlessly obfuscated code. If you also think about writing code like writing literature (yes, I know they are different things), design patterns would be the loose equivalent of filling your prose with other people's paragraphs, which takes all the fun out of it. Design your own shit (unless you are stuck), it's a lot more fun. The great thing about doing it this way is as you gain more experience, you will independently arrive at some of the design patterns, and that will feel a lot better than just mindlessly trying to insert them into your code everywhere.
I actually wouldn't disagree about it being verbal diarreah. I just sort of threw everything out there in a rant style (since it was late) rather than taking the time to really structure everything, remove fluff sentences, and generally clean up my post. Which is ironic given that it's about clean code.
Design patterns can indeed be extremely good solutions. As can established algorithms. But as you rightly point out, whichever design pattern is popular at the time will be the "correct" way to program for a lot of people, until a new one becomes popular, then they forget the old one, and consider everything made with the old one to be "unmaintainable and out of date legacy code", because they never wrote good code to begin with. Nowhere is this more true than in the web space (I am convinced that 100% of web developers are completely incompetent if not outright frauds, but that's a rant for another day), where there seems to be a new "revolutionary" javascript framework every other week, and it always comes with some "new" design paradigm that just ends up being more boilerplate for no real benefit.
Comparing code to literature isn't that different after all. There's a reason for the old adage "good code should read like good prose".
And yeah, everyone is expected to make mistakes and write shitty code from time to time. Coding is hard, inherently, and nobody (even the experts) is particularly good at it. Humans are just not good at coding. But you can look for smells. One of them (which is usually considered a feature, not a smell) is when you have a IDelegateClassInstancerFactory or some other meaningless class names littering your codebase. If it's not a real concept with a real responsibility, ditch it. Most design patterns (and especially anti-patterns) will encourage the opposite. It's why I hate service-locator and singleton so much (seriously, NEVER use singletons. Not even in rare exceptions. They are literally completely useless in 100% of cases. There are no niches or special cases, just avoid them at all costs. They are kryptonite to good code. Even if people only skim this thread and don't bother learning anything, if they at least get the gist and as a result one time for one random project decide not to use a singleton one time, I will consider all of this worth it.)