Googling around and reading the first day the following stands out:
- Type inference with the safety of static typing (we're all sick of typing types aren't we??)
- Companion objects instead of static methods
- Higher order functions (why do we need lambas? Can't we just switch to Scala?)
- Interfaces, no I mean mixins, wait traits: interfaces with implementation? I think these are great when used well.
- Tuples and ranges.
The exercise for day 1 is to write noughts and crosses. I decided to concentrate on setting up a nice environment for writing scala rather than great code. I started with vim, then sublime but decided I fancied using an IDE. I started with Eclipse but moved to IntelliJ as it looks nicer on a retina screen :) In addition to setting up an IDE I wanted to do TDD from the start, so I got to grips with ScalaTest.
Here are a couple of example tests:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
class BoardSuite extends FunSuite with BeforeAndAfter { | |
var board : Board = _ | |
before { | |
board = new Board | |
} | |
test("A board is initialized to blank") { | |
board.moves.foreach(row => row.foreach(box => assert(box === Player.B))) | |
} | |
test("Making the first move is marked X") { | |
board.move(1, 1) | |
assert(board.moves(1)(1) === Player.x) | |
} | |
test("Making the second move is marked o") { | |
board.move(1, 1) | |
board.move(1, 2) | |
assert(board.moves(2)(1) === Player.o) | |
} | |
test("Cant use the same place twice") { | |
board.move(1, 1) | |
intercept[RuntimeException] { | |
otherPlayersMove(1, 1) | |
} | |
} | |
test("No winner after no moves") { | |
val winner = board.winner() | |
println(board) | |
assert(winner === Player.B) | |
} | |
test("Win vertical line on left") { | |
board.move(0,0) | |
otherPlayersMove(1,0) | |
board.move(0,1) | |
otherPlayersMove(1,1) | |
board.move(0,2) | |
println(board) | |
assert(Player.x === board.winner) | |
} |
Here is the main Board object. I could pull out the logic to determine a winner into a class like Game but as this was my first Scala code I decided to keep it simple in a single file.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
package com.batey.sevenlanguages.scala.dayone | |
object Player extends Enumeration { | |
type Player = Value | |
val x, o, B = Value | |
} | |
import com.batey.sevenlanguages.scala.dayone.Player._ | |
class Board() { | |
val size = 3 | |
var moves = Array.ofDim[Player](size,size) | |
var nextMove = Player.x | |
moves.foreach(arg => { | |
for (i <- 0 until arg.length) { | |
arg(i) = Player.B | |
} | |
}) | |
override def toString() : String = { | |
val toReturn = new StringBuilder | |
for (i <- 0 until size) { | |
moves(i).foreach(col => toReturn ++= col + " ") | |
toReturn ++= "\n" | |
} | |
toReturn.toString | |
} | |
def move(x:Int, y:Int) = { | |
if (moves(y)(x) != Player.B) { | |
throw new RuntimeException("Move already made") | |
} | |
moves(y)(x) = nextMove | |
if (nextMove == Player.x) { | |
nextMove = Player.o | |
} else { | |
nextMove = Player.x | |
} | |
} | |
def winner() : Player = { | |
var win = horizontalWin | |
if (win != Player.B) return win | |
win = verticalWin | |
if (win != Player.B) return win | |
win = diagWinFromTopLeft | |
if (win != Player.B) return win | |
diagWinFromTopRight | |
} | |
} |
I've omitted the code that calculates the different types thought it's all available on github. I looked at using a case class rather than a class that extends Enumeration but I haven't been taught case classes yet so I'll leave that until later.
That's it for day one!
No comments:
Post a Comment