Code golf is a quirky little game – complete the challenge at hand in your preferred programming language in as few bytes as possible. Whole languages exist just for this purpose, with single-character commands and little regard for whitespace.
dc always seemed like a plausible language for these exercises, and I recently attempted a few tasks which I would not ordinarily use dc for, notably 99 Bottles of Beer.
It’s a simple enough task – count down from 99, perform some string concatenation, and be done with it. But a bit of a spanner is thrown into the works in that last verse – suddenly, you need to factor in pluralization; you need to change the 2nd line; and you need to restart the counter at 99:
[…] 3 bottles of beer on the wall, 3 bottles of beer. Take one down and pass it around, 2 bottles of beer on the wall. 2 bottles of beer on the wall, 2 bottles of beer. Take one down and pass it around, 1 bottle of beer on the wall. 1 bottle of beer on the wall, 1 bottle of beer. Go to the store and buy some more, 99 bottles of beer on the wall.
My initial attempt was 269 bytes long:
[ bottle]sa[[s]P]sb[ of beer]sc[ on the wall]sd[, ]se[.]sf10sg[[Take one down and pass it around]Pq]sh99si[ll1
It used two counters (
i) and a lot of wasteful negotiating between the two in order to balance out the pluralization and beer count reset. A couple of bytes could be shaved off here and there – use the stack instead of a variable load for the decrement loop conditional, use the ASCII value 46 instead of the three-character string
[.], but ultimately the design was flawed. Refactoring brought me where I am now, 237 bytes:
- Initialize our counter i to 299:
- m decrements the counter, duplicates that value (we need it at the end to check if our loop should keep going), divides it by 3, checks our counter, and runs v to add 99 if we’re at our final print. It prints (number of beers) bottle, and runs n to print “s” if we have multiple beers:
- n is our pluralizer, all it does is print “s”:
- Just a string:
[ of beer]so
- Just a string:
[ on the wall]sp
- Just a string:
- Save a byte using ASCII instead of a string (period):
- Save a byte using ASCII instead of a string (linefeed):
- t is largely just a printed string, but the quit command acts as an else in conjunction with the macro (u) that calls it:
[[Take one down and pass it around]Pq]st
- u runs t if we’re not on the last verse. It would then print the ‘Go to the store’ string, except that we used that quit command to back out:
[d4<t[Go to the store and buy some more]P]su
- v magically produces 99 additional beers. Nice!:
- Our main loop runs the printing macros, prints the simple strings, checks the counter and runs itself if we still have song to sing. The trailing x command starts the loop running:
Decreasing the byte count (golfing the code, so to speak) required getting rid of as much bulky counter manipulation as possible. dc starts with a default precision of 0 places, so we can use one counter decrementing by one every time from 299 to give us two values: how many beers (i/3) and which print operation we’re on (i).
dc as a golf language, then
dc seems like it would make for a good golf language – it is stack-based, all commands are a single letter, and whitespace is generally unnecessary. Yet far more verbose languages have far smaller answers, largely because dc’s command set is so limited. Everything string-related is a stack operation; there are no string manipulation abilities. Strings aren’t even truly concatenated, they’re just printed without newlines. There are 17 printing commands1 in those 237 bytes. I could theoretically do some of this printing in a loop, but I need so many conditionals that various
P loops would likely add as much as it golfs. Not to mention ordering would rotate back and forth because of the stack.
Speaking of the stack, one might think there’s efficiency to be gained via stack manipulation, but there just aren’t many commands for that either. It’s a theoretically infinite stack (like RPL), so there’s no concept of rolling over, and there’s no RPN z level for repetition. There’s no picking (arbitrary roll), nor is there even a standalone drop. You basically do things that push and pop, and the only manipulation you get is a single-level swap (
reverse). dc expects you to use registers.
Decrementing (or incrementing) a counter is quite wasteful. You need to load it onto the stack,
li, decrement it,
1-, duplicate the value (because presumably you need it for something and
d now is shorter than
d, and then store the new value,
Two things hadn’t occurred to me before this exercise – it’s smaller (albeit less readable) to store a single-character string low on the ASCII table with its value; and a byte can be shaved off of storing and immediately running a macro a by duplicating vs. storing and loading (
[1+]salax). Not particularly useful in day-to-day life, but good for the ol’ brain.
dc is certainly not intended as a full-featured language (it is, after all, just the ‘desk calculator’). Nor was it designed with ‘golfing’ in mind, and its limitations are quite apparent here. But it’s still my go-to calculator on the CLI,
abusing it in this manner makes for a good exercise, and the end result wasn’t half bad.
- dc has 4 printing commands, two of which (
f) insert their own newlines. I only use
n, the former for strings and ASCII values; the latter for numbers. ↩︎