Skip to content

Commit 57ec35f

Browse files
committed
Add 07: Magic square solver
Initial dumb implementation. - CPU profiling. To see the result, run 07-magic_square, execute `go tool pprof 07-magic_square path/to/cpu.pprof` and input `top[Enter]` in the prompt. - Generic algorithm. The trick is the Float64Slice adaptor; see [1]. Maybe I will use the technique used in the sort library[2][3] which allows simpler API. [1]: http://stackoverflow.com/a/6256127/5266681 [2]: golang/go#16721 [3]: golang/go@22a2bdf
1 parent d9860d4 commit 57ec35f

11 files changed

+329
-0
lines changed

07-magic_square/main.go

+36
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,36 @@
1+
package main
2+
3+
import (
4+
"fmt"
5+
"math/rand"
6+
"os"
7+
"time"
8+
9+
"github.com/frickiericker/learn-go/07-magic_square/matrix"
10+
"github.com/pkg/profile"
11+
)
12+
13+
func main() {
14+
if err := run(); err != nil {
15+
fmt.Fprintf(os.Stderr, "error: %s\n", err)
16+
os.Exit(1)
17+
}
18+
}
19+
20+
func run() error {
21+
defer profile.Start(profile.CPUProfile).Stop()
22+
23+
n := 5
24+
rnd := rand.New(rand.NewSource(time.Now().UnixNano()))
25+
solver := NewMagicSquareSolver(n, rnd)
26+
solver.Randomize()
27+
28+
for {
29+
if solver.SearchNeighbor() == 0 {
30+
break
31+
}
32+
}
33+
matrix.Print(solver.Get())
34+
35+
return nil
36+
}

07-magic_square/matrix/interface.go

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
package matrix
2+
3+
type Interface interface {
4+
Rows() int
5+
Cols() int
6+
Get(i, j int) float64
7+
Set(i, j int, value float64)
8+
}

07-magic_square/matrix/io.go

+17
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
package matrix
2+
3+
import (
4+
"fmt"
5+
)
6+
7+
func Print(mat Interface) {
8+
for i := 0; i < mat.Rows(); i++ {
9+
for j := 0; j < mat.Cols(); j++ {
10+
if j > 0 {
11+
fmt.Print("\t")
12+
}
13+
fmt.Printf("%.3g", mat.Get(i, j))
14+
}
15+
fmt.Print("\n")
16+
}
17+
}

07-magic_square/matrix/square.go

+41
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
package matrix
2+
3+
type Square struct {
4+
data []float64
5+
size int
6+
}
7+
8+
func NewSquare(size int) *Square {
9+
return &Square{
10+
data: make([]float64, size*size),
11+
size: size,
12+
}
13+
}
14+
15+
func (mat *Square) Data() []float64 {
16+
return mat.data
17+
}
18+
19+
func (mat *Square) Size() int {
20+
return mat.size
21+
}
22+
23+
func (mat *Square) Rows() int {
24+
return mat.size
25+
}
26+
27+
func (mat *Square) Cols() int {
28+
return mat.size
29+
}
30+
31+
func (mat *Square) locate(i, j int) int {
32+
return i*mat.size + j
33+
}
34+
35+
func (mat *Square) Get(i, j int) float64 {
36+
return mat.data[mat.locate(i, j)]
37+
}
38+
39+
func (mat *Square) Set(i, j int, value float64) {
40+
mat.data[mat.locate(i, j)] = value
41+
}

07-magic_square/matrix/square_test.go

+39
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,39 @@
1+
package matrix
2+
3+
import (
4+
"testing"
5+
)
6+
7+
func TestNewSquare_checkSize(t *testing.T) {
8+
sq := NewSquare(7)
9+
if result := sq.Size(); result != 7 {
10+
t.Errorf("unexpected result from Size() %v", result)
11+
}
12+
if result := sq.Rows(); result != 7 {
13+
t.Errorf("unexpected result from Rows() %v", result)
14+
}
15+
if result := sq.Cols(); result != 7 {
16+
t.Errorf("unexpected result from Cols() %v", result)
17+
}
18+
if result := sq.Data(); len(result) != 7*7 {
19+
t.Errorf("unexpected result from Data() %v", result)
20+
}
21+
}
22+
23+
func TestSquare_Get_safeRange(t *testing.T) {
24+
sq := NewSquare(5)
25+
for i := 0; i < 5; i++ {
26+
for j := 0; j < 5; j++ {
27+
sq.Get(i, j)
28+
}
29+
}
30+
}
31+
32+
func TestSquare_Set_safeRange(t *testing.T) {
33+
sq := NewSquare(5)
34+
for i := 0; i < 5; i++ {
35+
for j := 0; j < 5; j++ {
36+
sq.Set(i, j, 1.23)
37+
}
38+
}
39+
}

07-magic_square/seq/interface.go

+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
package seq
2+
3+
type Interface interface {
4+
Len() int
5+
}
6+
7+
type Swappable interface {
8+
Interface
9+
Swap(i, j int)
10+
}

07-magic_square/seq/shuffle.go

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
package seq
2+
3+
import (
4+
"math/rand"
5+
)
6+
7+
func Shuffle(seq Swappable, rnd *rand.Rand) {
8+
for i := seq.Len() - 1; i > 0; i-- {
9+
j := rnd.Intn(i + 1)
10+
seq.Swap(i, j)
11+
}
12+
}

07-magic_square/seq/shuffle_test.go

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
package seq
2+
3+
import (
4+
"math/rand"
5+
"testing"
6+
)
7+
8+
func TestShuffle_isPermutation(t *testing.T) {
9+
rnd := rand.New(rand.NewSource(0))
10+
seq := Float64Slice{[]float64{1, 2, 3, 4}}
11+
Shuffle(seq, rnd)
12+
13+
counter := map[float64]int{}
14+
for _, value := range seq.slice {
15+
counter[value] += 1
16+
}
17+
for _, count := range counter {
18+
if count != 1 {
19+
t.Errorf("%v", seq)
20+
}
21+
}
22+
}
23+
24+
func TestShuffle_allowEmptySeq(t *testing.T) {
25+
rnd := rand.New(rand.NewSource(0))
26+
seq := Float64Slice{[]float64{}}
27+
Shuffle(seq, rnd)
28+
}

07-magic_square/seq/slice.go

+17
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
package seq
2+
3+
type Float64Slice struct {
4+
slice []float64
5+
}
6+
7+
func NewFloat64Slice(slice []float64) Float64Slice {
8+
return Float64Slice{slice}
9+
}
10+
11+
func (seq Float64Slice) Len() int {
12+
return len(seq.slice)
13+
}
14+
15+
func (seq Float64Slice) Swap(i, j int) {
16+
seq.slice[i], seq.slice[j] = seq.slice[j], seq.slice[i]
17+
}

07-magic_square/seq/slice_test.go

+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
package seq
2+
3+
import (
4+
"reflect"
5+
"testing"
6+
)
7+
8+
func TestFloat64Slice_Len(t *testing.T) {
9+
seq := Float64Slice{[]float64{1, 2, 3, 4}}
10+
if seq.Len() != 4 {
11+
t.Errorf("%v", seq)
12+
}
13+
}
14+
15+
func TestFloat64Slice_Swap(t *testing.T) {
16+
seq := Float64Slice{[]float64{1, 2, 3, 4}}
17+
seq.Swap(1, 2)
18+
if !reflect.DeepEqual(seq.slice, []float64{1, 3, 2, 4}) {
19+
t.Errorf("%v", seq)
20+
}
21+
}

07-magic_square/solver.go

+100
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,100 @@
1+
package main
2+
3+
import (
4+
"math"
5+
"math/rand"
6+
7+
"github.com/frickiericker/learn-go/07-magic_square/matrix"
8+
"github.com/frickiericker/learn-go/07-magic_square/seq"
9+
)
10+
11+
type MagicSquareSolver struct {
12+
square *matrix.Square
13+
magic float64
14+
rnd *rand.Rand
15+
}
16+
17+
func NewMagicSquareSolver(size int, rnd *rand.Rand) *MagicSquareSolver {
18+
square := matrix.NewSquare(size)
19+
for i := range square.Data() {
20+
square.Data()[i] = float64(i + 1)
21+
}
22+
return &MagicSquareSolver{
23+
square: square,
24+
magic: float64(magicConstant(size)),
25+
rnd: rnd,
26+
}
27+
}
28+
29+
func (solver *MagicSquareSolver) Get() *matrix.Square {
30+
return solver.square
31+
}
32+
33+
func (solver *MagicSquareSolver) Randomize() {
34+
seq.Shuffle(seq.NewFloat64Slice(solver.square.Data()), solver.rnd)
35+
}
36+
37+
func (solver *MagicSquareSolver) evaluate() float64 {
38+
loss := float64(0)
39+
magic := solver.magic
40+
for i := 0; i < solver.square.Size(); i++ {
41+
loss += math.Abs(sumRow(solver.square, i) - magic)
42+
loss += math.Abs(sumCol(solver.square, i) - magic)
43+
}
44+
loss += math.Abs(sumDiag(solver.square) - magic)
45+
loss += math.Abs(sumAntidiag(solver.square) - magic)
46+
return loss
47+
}
48+
49+
func (solver *MagicSquareSolver) SearchNeighbor() float64 {
50+
n := solver.square.Size()
51+
data := solver.square.Data()
52+
c1 := solver.rnd.Intn(n * n)
53+
c2 := solver.rnd.Intn(n * n)
54+
55+
lossBefore := solver.evaluate()
56+
data[c1], data[c2] = data[c2], data[c1]
57+
lossAfter := solver.evaluate()
58+
if lossAfter - 3 > lossBefore {
59+
data[c1], data[c2] = data[c2], data[c1]
60+
return lossBefore
61+
}
62+
return lossAfter
63+
}
64+
65+
func sumRow(square *matrix.Square, row int) float64 {
66+
sum := float64(0)
67+
for col := 0; col < square.Cols(); col++ {
68+
sum += square.Get(row, col)
69+
}
70+
return sum
71+
}
72+
73+
func sumCol(square *matrix.Square, col int) float64 {
74+
sum := float64(0)
75+
for row := 0; row < square.Rows(); row++ {
76+
sum += square.Get(row, col)
77+
}
78+
return sum
79+
}
80+
81+
func sumDiag(square *matrix.Square) float64 {
82+
sum := float64(0)
83+
for i := 0; i < square.Size(); i++ {
84+
sum += square.Get(i, i)
85+
}
86+
return sum
87+
}
88+
89+
func sumAntidiag(square *matrix.Square) float64 {
90+
sum := float64(0)
91+
size := square.Size()
92+
for i := 0; i < size; i++ {
93+
sum += square.Get(i, size-i-1)
94+
}
95+
return sum
96+
}
97+
98+
func magicConstant(n int) int {
99+
return n * (n*n + 1) / 2
100+
}

0 commit comments

Comments
 (0)