-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmain.go
109 lines (88 loc) · 2.63 KB
/
main.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
package main
import (
"fmt"
"math"
"math/rand"
"sort"
"sync"
)
const (
// The number of samples for each C-value. Smaller = fast, larger = better approximation
EV_SAMPLES = 200
// The precision of C-values. Smaller = fast, larger = more data
PRECISION = 3
// The C-value that the simulation will start at, as well as the amount the C-value
// is incremented by each iteration. The number of iterations needed starts to increase
// dramatically for C-values smaller than 1e-07
INITIAL_C_VALUE = 1e-08
)
// Takes a C-value as the initial probability, and simulates a "dice roll".
// If the roll succeeds, the function returns the number of dice rolls that happened.
// If the roll doesn't succeed, the probability increases by C, and another dice roll occurs.
// Eventually, the probability will exceed 1.0, and the dice roll is forced to succeed.
func sampleCValue(C float64) int64 {
if C <= 0 {
panic("C must be greater than zero")
} else if C >= 1 {
return 1
}
N := int64(1)
for {
if rand.Float64() <= C*float64(N) {
return N
}
N += 1
}
}
// Approximates the expected value (EV) for a given C-value.
func calcEV(C float64) float64 {
sum := int64(0)
for i := 0; i < EV_SAMPLES; i++ {
sum += sampleCValue(C)
}
return float64(sum) / float64(EV_SAMPLES)
}
func main() {
type Result struct {
C float64
P float64
EV float64
}
results := []Result{}
resultMutex := sync.Mutex{}
resultWaitGroup := sync.WaitGroup{}
nextStepIncrease := int(math.Pow(10, float64(PRECISION)))
i := 1
iStep := 1
C := float64(0)
// Computes the expected value for a range of C-values. The precision of the
// C-value is controlled in order to obtain a wide range of datapoints in a
// reasonable amount of time
for C < 1 {
C = INITIAL_C_VALUE * float64(i)
// The precision of the next C-value will be too high
if i == nextStepIncrease {
// Increasing the step by a factor of 10 shifts everything over one digit
iStep *= 10
// After 10 iterations a new digit will be introduced (1, 2, 3, ..., 9, 10)
// and the precision will need to be adjusted again
nextStepIncrease *= 10
}
resultWaitGroup.Add(1)
// Run the EV calc in parallel
go func(C float64, i int) {
defer resultWaitGroup.Done()
ev := calcEV(C)
resultMutex.Lock()
results = append(results, Result{C, 1 / ev, ev})
resultMutex.Unlock()
}(C, i)
i += iStep
}
resultWaitGroup.Wait()
// Sort results by C-value. They may be out of order due to running in parallel
sort.Slice(results, func(i, j int) bool { return results[i].C < results[j].C })
for _, row := range results {
fmt.Printf("%.12f,%.12f,%f\n", row.C, row.P, row.EV)
}
}