This repository has been archived on 2017-04-03. You can view files and clone it, but cannot push or open issues or pull requests.
blog_post_tests/20090507223612.blog
Olivier DOSSMANN d897ae448f Initial commit
2014-11-19 16:42:25 +01:00

9 lines
2.6 KiB
Plaintext

Engage the balonium generator, Scotty!
<p>I enjoy a game called &#8220;Commands and Colors: Ancients&#8221;, which I&#8217;ve blogged about here before. It&#8217;s a simulation of tactical ancient warfare that uses special dice to resolve battles. In any given battle you can consider each six-sided die to have faces labeled with the symbols Miss, Miss, Hit, Sword, Helmet, Flag. To improve my play, I decided to generate and study a table of the odds of getting a specified number of hits when a specified number of dice is rolled. I set out to write a Python program to do this.</p>
<p>There are special circumstances under which flags and helmets convert to hits, so the program actually needs to print out several tables and is not entirely trivial. Still, it is computing on a mathematically simple model with strictly bounded computational cost &#8211; except for war elephants. These units have the special ability that when they roll a sword, the sword is counted as a hit <em>and then rerolled</em>. (This may make more sense if you think of a sword roll as representing impact damage.) You keep rolling and marking hits as long as the die keeps coming up swords.</p>
<p>To calculate the non-elephant probabilities I enumerated eight entire state spaces for each of 1 to 8 dice (the most you can ever roll under the rules &#8211; takes Alexander the Great or Julius Caesar leading a Heavy unit with the the Clash of Shields +2 bonus active), then counted up instances of each distinct outcome (so many misses, hits, swords, flags, banners) to assign a probability mass to each. </p>
<p>(Statisticians often think in terms of probability mass or probability weight, which has to be conserved as a distribution changes. It&#8217;s analogous to thinking of electricity as a fluid.)</p>
<p>I then had to write code to mutate a copy of each distribution according to the elephants&#8217; sword-reroll rule. To do this, each outcome containing a sword hit needs its probability mass divided by six and reallocated to itself and five other outcomes with one fewer sword apiece; you stop reallocating when the probability mass on a sword-containing outcome drops below a very low noise level.</p>
<p>That particular piece of code gave me more trouble than the rest of the program put together. At one point I grumbled to a friend who had been following the project &#8220;I&#8217;m having persistent bugs in my probability mass reallocator.&#8221; </p>
<p>He looked right back at me and said, with a perfectly straight face, &#8220;Have you considered reversing the polarity of the neutron flow?&#8221;</p>