Against Abstractions in Programming

"Abstracting" is the term I will use for when a coder encapsulates some logic in a function purely for the purpose of summarizing it and giving it a simpler interface so that, ideally, the caller of said function doesn't have to care about the internal logic. I'm going to argue that this is bad, or at least has some negative consequences that people aren't adequately considering.

There are cases when encapsulating logic in a function is necessary and desirable, of course. For example, if you find yourself using identical logic repeatedly. In order to compress the code, you may want to put that logic in a function. This will reduce line count and minimize copy-paste errors.

Delegating work to a function also helps the coder mentally compress what the code is doing. It can be a lot easier to think about a long program if you've broken it up into several steps, each with discriptive names that represent the essence of what's going on.

Abstracting your code into a library will take this to the next level. You can encapsulate thousands of lines of code, maybe hundreds of thousands, and wrap it all up in a simple interface. By only requiring the caller to specifiy a few pieces of information, they can essentially get all the work you did for very little effort, drastically reducing their development time.

Now for the costs.

A function used in multiple places will be harder to change in the future, because to modify it safely you must consider every context it's used in.

Thinking about the whole program is actually extremely important. The program will change over time, and the steps you broke it into before may no longer be appropriate. Having broken the program into the wrong steps means it'll be harder to merge the steps together again. Or, you may elect to leave the code as it is as much as possible, and start introducing weird workarounds and redundant logic.

The more abstract your code, the more situations and use cases it needs to be able to handle. This implies that the user of said code is paying a cost. The user only needs to solve their specific use case, but the abstract code needs to handle many use cases, which very often is not done with a simple switch statement. This consumes more CPU and memory resources and makes the abstract code harder to understand...

Now, supposedly, abstract code is supposed to guarantee the user that they don't have to think about or understand the code being encapsulated, but that's rarely the case. Most libraries are really bad about checking input and providing clear error messages, for one thing. And for another thing, quite often the user does have to care about how exactly the underlying code works if they're trying to solve a hard problem. For example, something performance-intensive, or that has to run in multiple threads.

And finally, another huge problem is the long-term intellectual cost of incentivizing programming professionals to prioritize memorizing trivia. Trivia meaning the parts of programming that aren't concerned with the base reality of the problem being solved. And it's a problem inherent to abstractions. An abstraction has this feature where the user just provides the required fields. If the user doesn't have to understand what's going on within, they're essentially just filling out a form. By creating an abstraction, you're creating trivia. In the extreme case, programming will become nothing but managing configuration files. Not only is that, in my personal opinion, not very interesting, it's also not a transferrable skill. Libraries don't last very long. They go in and out of popularity all the time. Put another way: if you learn how to use, e.g., a networking library. You're not learning networking. The proof of that being: if that library goes away, you won't be able to do networking anymore.

I understand that many programmers don't much care about understanding how things really work, they just want a job. And it's easy to believe that we'll all always be able to find jobs where we continue to be valued for our knowledge of all the big libraries. Leave the low-level stuff to someone else. Especially since, even if you were interested in the low-level stuff, you're not able to apply it at work!

What's worse is, in my experience the "low-level" solution to problems is actually much simpler than the abstract solution. Of course, that's impossible to convince people of if they've only done programming through abstractions for several years, already have a set of tools they're comfortable with, and a career built on using them.

But, I have noticed the definition of "low-level" drifting upwards over time...

If you use Wordpress or React to make a website - have you ever tried making a website from scratch? You know, JavaScript, HTML, CSS?

If your server is Bottle or Node.js - have you ever tried writing a sever from scratch? You know, with a socket, behaving according to the RFC document for your chosen protocol?

I guess my point is that so few people now have had the experience of trying to solve these problems the low-level way. If companies had adopted a strategy of writing everything from scratch every time, sure, they'd probably end up making some tools, but I'm pretty sure they'd make different tools than the ones you're stuck with now, and you'd be surrounded by people who know real stuff instead of trivia.

I also worry that, eventually, there may not be a "someone else" who understands the "low-level" stuff at all.

I would say I fall into the spectrum of developer you're describing, I think for most functions someone abstracts there really is no intent to reuse them but they can be made local or private functions. That's really the whole idea with public and private methods. I like the idea of learning low level networking but it's rarely something I even need to think twice about and no customer is going to pay you to redevelop it, that's what it really comes down to. I agree though that most libraries don't help solve customer problems, especially things like react where most problems I've had and seen other people have aren't solving problems for the customer but with actually writing the code. I'd even go as far as saying functional programming doesn't solve customer problems. The best libraries just transparently solve problems without thinking much about the library itself, ie, a lot of the dotnet framework is like that. I think that's really the crux of what you're getting at, most libraries aren't that good and that's why they change so often. Especially Javascript is riddled with awful things.

Like
Reply

To view or add a comment, sign in

More articles by Jeffrey Spreng

Insights from the community

Others also viewed

Explore topics