In this letter, Dijkstra talks about readability and maintainability in a time where those topics were rarely talked about (1968). This letter was one of the main causes why modern programmers don’t have to trouble themselves with goto statements. Older languages like Java and C# still have a (discouraged) goto statement, because they (mindlessly) copied it from C, which (mindlessly) copied it from Assembly, but more modern languages like Swift and Kotlin don’t even have a goto statement anymore.
switch statements are three gotos in a trenchcoat.
Anything is a goto in disguise when you think about it
In most CPU instruction sets, the only conditional instruction is branch (aka goto).
Languages don’t have goto because they mindlessly copied it.
TIL that C# and Java have a goto statement.
Forget about goto, the latest version of C# introduced what’s essentially comefrom.
Please don’t link to medium articles. That page is terrible to visit.
To be fair,
await
is a bit more likecomefrom
, and it’s been around for a few releases now.async/await was introduced in version 4.5, released 2012. More than a few releases at this point!
Java doesn’t. Well, it’s a reserved keyword but it’s not implemented.
Yeah but we got labels with continue and break, so we can pseudo goto.
Following that logic
if
,else
andwhile
are also “pseudo goto” statements.There’s nothing wrong with conditional jumps - we couldn’t program without them. The problem with goto specifically is that you can goto “anywhere”.
If pretty much gets compiled to a goto statement. Well more a jumpif but same principle
For C it makes sense. The point of C is that it can work as a low level language. Basically, everything doable with assembly SHOULD be doable with C, and that’s why we don’t need another low level language that’s basically C with goto.
Even though almost all of C users should never use goto.
C is one of the few languages where using
goto
makes sense as a poor man’s local error/cleanup handler.Yeah. Without a proper error handling mechanism, goto is actually useful for once.
Still don’t get why Go simultaneously picked this and introduced
defer
For such an influential letter, I don’t find his arguement all that compelling. I agree that not using
go to
will often lead to better structured (and more maintainable) programs, but I don’t find his metric of “indexable process progress” to satisfyingly explain why that is.Perhaps it’s because at that time people would be running the programs in their heads before submitting them for processing, so they tended to use more of a computer scientist mindset - whereas now we’re more likely to use test cases to convince ourselves that code is correct.
I think it’s convoluted way to describe it. Very technically-practical. I agree it’s probably because of historical context.
The argument I read out of it is that using
goto
breaks you being able to read and follow the code logic/run-logic. Which I agree with.Functions are similar jumps, but with the inclusion of a call stack, you can traverse and follow them.
I think we could add a goto stack / include goto jumps in the call stack though? It’s not named though, so the stack is an index you only understand when you look at the code lines and match goto targets.
I disagree with unit tests replacing readability. Being able to read and follow code is central to maintainability, to readability and debug-ability. Those are still central to development and maintenance even if you make use of unit tests.
Perhaps it’s because at that time people would be running the programs in their heads before submitting them for processing, so they tended to use more of a computer scientist mindset - whereas now we’re more likely to use test cases to convince ourselves that code is correct.
This is 1968. You didn’t have an IDE or debugger. Your editor was likely pretty terrible too (if you had one). You may have still been using punch cards. It’s possible the only output you got from your program was printed on green-bar paper.
“Back in the day” it wasn’t uncommon to sit with a printout of your code and manually walk though it keeping state with a pencil. Being able to keep track of things in your head was very important.
GOTO existed in part for performance purposes. When your CPU clock is measured in megahertz, your RAM is measured in kilobytes and your compilers don’t do function in-lining it’s quicker and cheaper to just move the program pointer than it is to push a bunch of variables on a stack and jump to another location, then pop things off the stack when you’re done (especially if you’re calling your function inside a loop). Even when I was programming back in the '80s there was a sense that “function calls can be expensive”. It wasn’t uncommon then to manually un-roll loops or in-line code. Compilers are much more sophisticated today and CPUs are so much faster that 99% of the time you don’t need to think about now.
Oddly enough the arguments against GOTO are less important today as we have much better development tools available to us. Though I certainly won’t recommend it over better flow-control structures.
When your CPU clock is measured in megahertz, your RAM is measured in kilobytes
Ah yes, the good ol’ days when developers programmed for efficiency.
Mostly because they had to. Writing efficient code and easy-to-read code are often at odds with each other. I like being able to create lots of functions that can be called from a loop without needing to worry too much about function call overhead. I can prioritize readability for some time before I ever need to worry about performance.
I get that but it seems as though no one cares at all about efficiency these days.
Does the catchphrase “blazing fast” ring any bells? Some people care.
(Arguably that’s just the pendulum swinging the other way; Ruby, Python, and Java ruled the software world for a while, and I think that’s a large part of why the Go and Rust communities make such a big deal about speed.)
Java does not support goto.
And yet it’s still easy to write spaghetti code in Java. Just abuse inheritance. Where is this function implemented? No one knows but the compiler!
You mean
SpaghettiFactory()
That doesn’t make it spaghetti code though. In well-written OOP code you shouldn’t care where a function is implemented. The problem is a much too high level of abstraction. If your high level code is so abstract that it is only running tasks and handling messages there’s no way to write it in a way that prevents mistakes because you couldn’t possibly know what the actual implementations do.