-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrun_risk_a_lot.m
139 lines (126 loc) · 3.71 KB
/
run_risk_a_lot.m
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
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
function [win_pcts, avg_att_lefts, avg_def_lefts, att_devs, def_devs] = run_risk_a_lot(n_att, n_def, count, enabled, matrices)
# n_att: Number of attacking armies
# n_def: Number of defending armies
# count: Number of simulations to run
# enabled: A vector describing which matrices to test
# e.g., [1; 0; 0; 0; 0] will run against "should_roll.txt" and no others
# matrices: The decision matrices to use
if nargin < 5
matrices = { get_dynamic_roll_mat('should_roll.txt');
get_dynamic_roll_mat('should_roll_joost.txt')
get_dynamic_roll_mat('should_roll_ger.txt')
ones(6, 6) * 2;
ones(6, 6) };
end
if nargin < 4
enabled = ones(size(matrices));
end
if nargin < 3
count = 100;
end
n_methods = size(matrices);
attack_wins = zeros(n_methods);
avg_att_lefts = zeros(n_methods);
avg_def_lefts = zeros(n_methods);
att_devs = zeros(n_methods);
def_devs = zeros(n_methods);
for i = 1:size(matrices)(1)
if enabled(i)
[att_left, def_left] = run_with(n_att, n_def, count, matrices{i});
attack_wins(i) = size(att_left)(1);
if size(att_left) ~= 0
avg_att_lefts(i) = mean(att_left);
att_devs(i) = std(att_left);
else
avg_att_lefts(i) = NaN();
att_devs(i) = NaN();
end
if size(def_left) ~= 0
avg_def_lefts(i) = mean(def_left);
def_devs(i) = std(def_left);
else
avg_def_lefts(i) = NaN();
def_devs(i) = NaN();
end
end
end
win_pcts = attack_wins / count;
end
function [remaining_for_attack, remaining_for_defense] = run_with(n_att, n_def, count, mat)
remaining_for_attack = [];
remaining_for_defense = [];
for runs = 1:count
[result, remaining_for_victor] = run_risk(n_att, n_def, mat);
if result == 1
remaining_for_attack = [remaining_for_attack; remaining_for_victor];
else
remaining_for_defense = [remaining_for_defense; remaining_for_victor];
end
end
end
function mat = get_dynamic_roll_mat(fname)
f = fopen(fname);
c = textscan(f, '%d %d %f', 'CollectOutput', 1);
fclose(f);
mat = accumarray(c{1}, c{2});
end
function [result, remaining_for_victor] = run_risk(attack_count, defense_count, roll_mat)
while attack_count > 1 && defense_count > 0
if attack_count >= 4
if defense_count >= 2
[a_diff, d_diff] = battle_vs_2(3, roll_mat);
else
[a_diff, d_diff] = battle(3, 1);
end
elseif attack_count == 3
if defense_count >= 2
[a_diff, d_diff] = battle_vs_2(2, roll_mat);
else
[a_diff, d_diff] = battle(2, 1);
end
else # attack_count == 2
if defense_count >= 2
[a_diff, d_diff] = battle(1, 2);
else
[a_diff, b_diff] = battle(1, 1);
end
end
attack_count = attack_count + a_diff;
defense_count = defense_count + d_diff;
end
result = defense_count == 0;
if defense_count == 0
remaining_for_victor = attack_count;
else
remaining_for_victor = defense_count;
end
end
function [a_diff, d_diff] = battle(n_att, n_def)
smaller = min(n_att, n_def);
att = roll(n_att)(1:smaller);
def = roll(n_def)(1:smaller);
[a_diff, d_diff] = fight(att, def);
end
function [a_diff, d_diff] = battle_vs_2(n_att, roll_mat)
att = roll(n_att)(1:2);
def_num = roll_mat(att(1), att(2));
def = roll(def_num);
[a_diff, d_diff] = fight(att(1:def_num), def);
end
function arr = roll(n)
assert(n <= 3)
arr = sort(randi(6, n, 1), 'descend');
end
function [a_diff, d_diff] = fight(att, def)
assert(size(att) == size(def))
res = att > def;
a_diff = 0;
d_diff = 0;
for i = 1:size(res)(1)
if res(i)
d_diff = d_diff - 1;
else
a_diff = a_diff - 1;
end
end
end