Skip to content

Instantly share code, notes, and snippets.

@LeifW
Forked from jrudolph/TowersOfHanoi.scala
Last active August 29, 2015 14:05
Show Gist options
  • Save LeifW/c8303e25906954c8bf3b to your computer and use it in GitHub Desktop.
Save LeifW/c8303e25906954c8bf3b to your computer and use it in GitHub Desktop.
/*
* This is the Towers of Hanoi example from the prolog tutorial [1]
* converted into Scala, using implicits to unfold the algorithm at
* compile-time.
*
* [1] http://www.csupomona.edu/~jrfisher/www/prolog_tutorial/2_3.html
*/
object TowersOfHanoi {
import scala.reflect.Manifest
def simpleName(m:Manifest[_]):String = {
val name = m.toString
name.substring(name.lastIndexOf('$')+1)
}
sealed trait Nat
final class _0 extends Nat
final class Succ[Pre<:Nat] extends Nat
type _1 = Succ[_0]
type _2 = Succ[_1]
type _3 = Succ[_2]
type _4 = Succ[_3]
case class Move[N<:Nat,A<:Rod,B<:Rod,C<:Rod]()
implicit def move0[A<:Rod,B<:Rod,C<:Rod](implicit a:Manifest[A],b:Manifest[B]):Move[_0,A,B,C] = {
System.out.println("Move from "+simpleName(a)+" to "+simpleName(b));null
}
implicit def moveN[P<:Nat,A<:Rod,B<:Rod,C<:Rod](implicit m1:Move[P,A,C,B],m2:Move[_0,A,B,C],m3:Move[P,C,B,A])
:Move[Succ[P],A,B,C] = null
def run[N<:Nat,A<:Rod,B<:Rod,C<:Rod](implicit m:Move[N,A,B,C]) = null
sealed trait Rod
final class Left extends Rod
final class Center extends Rod
final class Right extends Rod
def main(args:Array[String]){
run[_2,Left,Right,Center]
}
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment