[net.unix] sed can do Towers of Hanoi

gbergman@ucbtopaz.CC.Berkeley.ARPA (11/21/84)

If you put the 60-line sed script near the end of this message
in a file, e.g. sed.hanoi, and then do
	sed -f sed.hanoi
and type in, say
	:abcd: : :<CR><CR>
(notice-- TWO carriage returns-- a peculiarity of sed), then
it will output the sequence of states involved in moving 4 rings,
the largest called "a" and the smallest called "d", from the
first to the second of three towers, so that the rings on any
tower at any time are in descending order of size.  You can start with
a different arrangement and a different number of rings, say
:ce:b:ax: and it will still give the shortest procedure for
moving them all to the middle tower.  The rules are: the names
of the rings must all be lower-case letters, they must be input within
3 fields (representing the towers) delimited by 4 colons, such that
the letters within each field are in alphabetical order (= rings in
descending order of size).
     For the benefit of anyone who wants to figure out the script,
I will explain that an "internal" line of the form
		b:0abx:1a2b3 :2   :3x2     
has the following meaning: the material after the three markers :1 :2
:3 represents the three towers; in this case the current set-up is
:ab :   :x  :.  The numbers after a, b and x in these fields indicate
that the next time it gets a chance, it will move a to tower 2, move b
to tower 3, and move x to tower 2.  The string after :0 just keeps
track of the alphabetical order of the names of the rings.  The b at the
beginning means that it is now dealing with ring  b  (either about to
move it, or re-evaluating where it should next be moved to).
     Although this version is "limited" to 26 rings because of the size
of the alphabet, one could write a script using the same idea in which
the rings were represented by arbitrary [strings][within][brackets], and
in place of the built-in line of the script giving the order of the
letters of the alphabet, it would accept from the user a line giving the
ordering to be assumed, e.g. [ucbvax][decvax][hplabs][foo][bar].
     While on the subject, I will repeat at the very end of this note
a much shorter file to do Towers of Hanoi using troff that a friend
posted for me a year or so ago, before I was on a machine that
subscribed to the net.  If you put it in a file "troff.hanoi", and
do  nroff -rr5 troff.hanoi,  it will output instructions for moving 5
rings from tower 1 to tower 2; generally, just put the desired number
of rings after the -rr on the command line.  In this case, you don't
have a choice of initial state, aside from choosing the number of rings.
			Have fun!
			George Bergman
			Math, UC Berkeley 94720 USA

		ucbvax!ucbcartan!gbergman (if your mailer can
			digest a 9-letter name; if not maybe:)
		ucbvax!cartan:gbergman  or
		gbergman%cartan@berkeley

#----------------------sed.hanoi------------------------------
# cleaning, diagnostics
s/  *//g
/^$/d
/[^a-z:]/{a\
Impermissible characters: use only a-z and ":".  Try again.
d
}
/^:[a-z]*:[a-z]*:[a-z]*:$/!{a\
incorrect format: use\
\	: string1 : string2 : string3 :<CR><CR>\
Try again.
d
}
/\([a-z]\).*\1/{a\
Repeated letters not allowed.  Try again.
d
}
# initial formatting
h
s/[a-z]/ /g
G
s/^:\( *\):\( *\):\( *\):\n:\([a-z]*\):\([a-z]*\):\([a-z]*\):$/:1\4\2\3:2\5\1\3:3\6\1\2:0/
s/[a-z]/&2/g
s/^/abcdefghijklmnopqrstuvwxyz/
:a
s/^\(.\).*\1.*/&\1/
s/.//
/^[^:]/ba
s/\([^0]*\)\(:0.*\)/\2\1:/
s/^[^0]*0\(.\)/\1&/
:b
# outputting current state without markers
h
s/.*:1/:/
s/[123]//gp
g
:c
# establishing destinations
/^\(.\).*\1:1/td
/^\(.\).*:1[^:]*\11/s/^\(.\)\(.*\1\([a-z]\).*\)\3./\3\2\31/
/^\(.\).*:1[^:]*\12/s/^\(.\)\(.*\1\([a-z]\).*\)\3./\3\2\33/
/^\(.\).*:1[^:]*\13/s/^\(.\)\(.*\1\([a-z]\).*\)\3./\3\2\32/
/^\(.\).*:2[^:]*\11/s/^\(.\)\(.*\1\([a-z]\).*\)\3./\3\2\33/
/^\(.\).*:2[^:]*\12/s/^\(.\)\(.*\1\([a-z]\).*\)\3./\3\2\32/
/^\(.\).*:2[^:]*\13/s/^\(.\)\(.*\1\([a-z]\).*\)\3./\3\2\31/
/^\(.\).*:3[^:]*\11/s/^\(.\)\(.*\1\([a-z]\).*\)\3./\3\2\32/
/^\(.\).*:3[^:]*\12/s/^\(.\)\(.*\1\([a-z]\).*\)\3./\3\2\31/
/^\(.\).*:3[^:]*\13/s/^\(.\)\(.*\1\([a-z]\).*\)\3./\3\2\33/
bc
# iterate back to find smallest out-of-place ring
:d
s/^\(.\)\(:0[^:]*\([^:]\)\1.*:\([123]\)[^:]*\1\)\4/\3\2\4/
td
# move said ring (right, resp. left)
s/^\(.\)\(.*\)\1\([23]\)\(.*:\3[^ ]*\) /\1\2 \4\1\3/
s/^\(.\)\(.*:\([12]\)[^ ]*\) \(.*\)\1\3/\1\2\1\3\4 /
tb
s/.*/Done!  Try another, or end with ^D./p
d
#------------------end sed.hanoi------------------------------

.\"------------------troff.hanoi----------------------------------
.de H
.nr d \\$1-1
.if \\nd .H \\nd \\$2 \\$4 \\$3
 Transfer ring \\$1 from tower \\$2 to tower \\$3.
.nr d \\$1-1
.if \\nd .H \\nd \\$4 \\$3 \\$2
..
Initial state:  \nr rings all on tower A; #1 on top.
.H \nr A B C
 Done!

perlman@wivax.UUCP (Gary Perlman) (11/26/84)

Very cute!

Note that the following line in the nroff version will
take care of a default (in this case, 5):
.	if \nr=0 .rn r 5

I've posted a pretty curses version of hanoi to net.sources.
Animation and all!

Gary Perlman/Wang Institute/Tyng Road/Tyngsboro, MA/01879/(617) 649-9731