Zig NEWS

Cover image for Convert decimal odds to probabilities with Zig
Vinícyus Brasil
Vinícyus Brasil

Posted on

Convert decimal odds to probabilities with Zig

Introduction

As sports betting expands across the world, more and more players don't understand tha math behind it. So, the purpose of this article it's to clarify how to convert from decimal odds to the probability behind them. The calculations are implemented in Zig, a system programming language perfect to create CLI tools because it's fast and simple.

This article it's divided into two sections: first we'll talk about the math and them about the implementation. If you only need the code you can go directly to my GitHub and this post was originally posted on my blog.

The math

A decimal odd is how much you're going to win for each dollar spent. For example, if the odd is 4.1, for each dollar you bet you're going to receive $4.1. Let's call the decimal odd oi o_i . The probability pi p_i related to the decimal odd oi o_i is

pi=1oi.p_i = \frac{1}{o_i}.

In the example, the probability related to the odd 4.1 is approximate 24.39%.

Let us think the other way now. Given an event with the probability of 20%, let's calculate how much the cassino should pay you:

0.2=1oi0.2 = \frac{1}{o_i} \therefore

oi=10.2=5.o_i = \frac{1}{0.2} = 5.

So, for each dollar spent, the cassino should be paying you 5 dollars, but it doesn't. The cassino only pays you a fraction of it, normally around 95%. The name of this fraction is payout. Naming it Equation 1, let us denote the payout by k k and write:

pi=k1oi.p_i = k * \frac{1}{o_i}.

Now calculate the odd with k=0.95k = 0.95 :
0.2=k1oi0.2 = k * \frac{1}{o_i} \therefore

oi=0.9510.2=4.75.o_i = 0.95 * \frac{1}{0.2} = 4.75.

Therefore, for each dollar spent, you're going to receive only 4.75 dollars. The 0.25 dollar here is the revenue of the cassino and that's how they can make money with this sorts of gambles.

When the odds are given by the cassino, the payout is unknown and it can vary depending on the time of day, the importance of that game in particular etc. So, we need to calculate first the payout and then we can convert and find the probabilities the cassino is using with Equation 1. Given that the sum of probabilities must be equal to 1 (by the Kolmogorov's axioms), we can write

1=i=1Npi1 = \sum_{i=1}^{N} p_i

where N is the number of possible outcomes of the game. For a football game (or soccer for my beloved american readers), the value of N is 3, where all the events are the home team win, the away team win or a draw. For a basketball game, N = 2 (the home team win or the away team win). Since pi=k1oi. p_i = k * \frac{1}{o_i}. , the sum of the probabilities can be written as

1=i=1Nk1oi.1 = \sum_{i=1}^{N} k * \frac{1}{o_i}.

Isolating kk , we have now Equation 2:
k=1i=1N1oik = \frac{1}{\sum_{i=1}^{N} \frac{1}{o_i}}

With the payout calculated we can now use the odds to calculate the probabilities.

Let's do an example and use the following match odds.

Printscreen from a sport betting website with the odds of 1.27 to Liverpool to win, 6.00 to a draw and 10.25 for Man United to win

Let o1 o_1 be the probability of Liverpool to win, o2 o_2 the probability of a draw and o3 o_3 be the probability of Man United to win. Calcutating the payout with Equation 2:

k=1i=1N1oik = \frac{1}{\sum_{i=1}^{N} \frac{1}{o_i}} \therefore

k=111.27+16.00+110.25k = \frac{1}{\frac{1}{1.27} + \frac{1}{6.00} + \frac{1}{10.25}} \therefore

k0.95091.k \approx 0.95091.

With the payout calculated, we can now calculate the probabilities using Equation 1.

  • Liverpool to win:

    p1=0.9509111.27p_1 = 0.95091 * \frac{1}{1.27} \therefore
    p1=74.8748%p_1 = 74.8748 \%
  • Draw:

    p2=0.9509116.00p_2 = 0.95091 * \frac{1}{6.00} \therefore
    p2=15.8485%p_2 = 15.8485 \%
  • Man United to win:

    p3=0.95091110.25p_3 = 0.95091 * \frac{1}{10.25} \therefore
    p3=9.2772%p_3 = 9.2772 \%

Zig implementation

The version of Zig used is 0.13 since it's the last stable release. The implementation must follow three steps:

1) Collect all the odds from the command line;
2) Calculate Equation 2 (payout) and then Equation 1 (the probabilities);
3) Print the values in the screen;

We can start by building an empty project

mkdir zigodd

zig init
Enter fullscreen mode Exit fullscreen mode

To get the arguments from the command line:

const std = @import("std");
const stdout = std.io.getStdOut().writer();

pub fn main() anyerror!void {

    const args = try std.process.argsAlloc(std.heap.page_allocator);
    defer std.process.argsFree(std.heap.page_allocator, args);
}
Enter fullscreen mode Exit fullscreen mode

A way of implementing the equations above it's creating a struct
that keeps the values of the payout, the odds and the probabilities.
Both of the later starts as empty arrays and the payout start with the value zero.
Two functions are incorporated in the struct: a method for calculating the payout
and other to calculate the probabilities.

const Odds: type = struct {
    k: f32 = 0.0,
    odds: []f32 = undefined,
    probabilities: []f32 = undefined,

    pub fn calculate_k(self: *Odds) anyerror!void {
        var sum: f32 = 0;

        for (self.odds) |item| {
            sum += (1 / item);
        }

        self.k = 1 / sum;
    }

    pub fn calculate_probabilities(self: *Odds) anyerror!void {
        const allocator = std.heap.page_allocator;
        var probs_processed = std.ArrayList(f32).init(allocator);

        for (self.odds) |item| {
            const prob_processed = self.k * (1 / item);
            try probs_processed.append(prob_processed);
        }

        self.probabilities = probs_processed.items;
    }
};
Enter fullscreen mode Exit fullscreen mode

Inside the main function, the odds from the arguments can be allocated
in a ArrayList and parsed into a float (f32). To convert to array,
just call the attribute .items.

    var myodds = Odds{};

    const len_probabilities: u32 = @intCast(args.len);

    const odds_raw = args[1..len_probabilities];

    const allocator = std.heap.page_allocator;
    var odds_processed = std.ArrayList(f32).init(allocator);

    defer odds_processed.deinit();

    for (odds_raw) |item| {
        const odd_processed = try std.fmt.parseFloat(f32, item);
        try odds_processed.append(odd_processed);
    }

    myodds.odds = odds_processed.items;
Enter fullscreen mode Exit fullscreen mode

Also inside the main function, we are now able to call the methods to calculate the values.

    try myodds.calculate_k();

    try myodds.calculate_probabilities();
Enter fullscreen mode Exit fullscreen mode

Finally, all the values can be printed to the screen.

    try stdout.print("-------ZIGODD: ODDS TO PROBABILITY CONVERSOR-------\n", .{});

    try stdout.print("---decimal odds to convert: \n", .{});

    for (myodds.odds, 0..) |item, index| {
        try stdout.print("o{d}: {d:0.2}\n", .{ index + 1, item });
    }
    try stdout.print("---calculated payout (k): \n", .{});
    try stdout.print("k: {d:0.5}\n", .{myodds.k});

    try stdout.print("---calculated probabilities: \n", .{});
    for (myodds.probabilities, 0..) |item, index| {
        try stdout.print("p{d}: {d:0.5}\n", .{ index + 1, item });
    }

    try stdout.print("-----------------------------------------------------\n", .{}); 

Note we are not worried about error handling. This should be a real concern to production apps, which is not our case here. As always, the full code is on my [GitHub](https://github.com/vinybrasil/zigodd).



That's it. Fell free to open a PR on the GitHub repo to contact me if you find something wrong. Keep on learning :D


Enter fullscreen mode Exit fullscreen mode

The project can be built with the following code:

zig build
Enter fullscreen mode Exit fullscreen mode

Running the code ./zig-out/bin/zigodd 1.27 6.00 10.25 with the values from the example give us the following result:

-------ZIGODD: ODDS TO PROBABILITY CONVERTER-------
---decimal odds to convert: 
o1: 1.27
o2: 6.00
o3: 10.25
---calculated payout (k): 
k: 0.9509055
---calculated probabilities: 
p1: 0.7487445
p2: 0.1584843
p3: 0.0927713
-----------------------------------------------------
Enter fullscreen mode Exit fullscreen mode

Note we are not worried about error handling. This should be a real concern to production apps, which is not our case here. As always, the full code is on my GitHub.

That's it. Fell free to open a PR on the GitHub repo to contact me if you find something wrong. Keep on learning :D

Top comments (0)