2022-07-05 15:55:47 +02:00
|
|
|
<?php
|
|
|
|
|
2024-02-02 15:16:11 +00:00
|
|
|
declare(strict_types=1);
|
|
|
|
|
2022-07-05 15:55:47 +02:00
|
|
|
namespace DNW\Skills;
|
2010-08-28 22:05:41 -04:00
|
|
|
|
|
|
|
// Container for a player's rating.
|
2022-07-05 15:33:34 +02:00
|
|
|
use DNW\Skills\Numerics\GaussianDistribution;
|
2016-05-24 14:10:39 +02:00
|
|
|
|
2022-07-05 16:21:06 +02:00
|
|
|
class Rating implements \Stringable
|
2010-08-28 22:05:41 -04:00
|
|
|
{
|
2023-08-01 14:02:12 +00:00
|
|
|
private const CONSERVATIVE_STANDARD_DEVIATION_MULTIPLIER = 3;
|
2010-08-28 22:05:41 -04:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Constructs a rating.
|
2022-07-05 15:55:47 +02:00
|
|
|
*
|
2023-08-02 09:15:01 +00:00
|
|
|
* @param float $mean The statistical mean value of the rating (also known as mu).
|
|
|
|
* @param float $standardDeviation The standard deviation of the rating (also known as s).
|
|
|
|
* @param float|int $conservativeStandardDeviationMultiplier optional The number of standardDeviations to subtract from the mean to achieve a conservative rating.
|
2010-08-28 22:05:41 -04:00
|
|
|
*/
|
2024-02-20 14:21:44 +00:00
|
|
|
public function __construct(private readonly float $mean, private readonly float $standardDeviation, private readonly float|int $conservativeStandardDeviationMultiplier = self::CONSERVATIVE_STANDARD_DEVIATION_MULTIPLIER)
|
2010-08-28 22:05:41 -04:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2023-08-02 09:04:56 +00:00
|
|
|
* The statistical mean value of the rating (also known as mu).
|
2010-08-28 22:05:41 -04:00
|
|
|
*/
|
2023-08-02 09:04:56 +00:00
|
|
|
public function getMean(): float
|
2010-08-28 22:05:41 -04:00
|
|
|
{
|
2023-08-02 09:15:01 +00:00
|
|
|
return $this->mean;
|
2010-08-28 22:05:41 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The standard deviation (the spread) of the rating. This is also known as s.
|
|
|
|
*/
|
2023-08-02 09:04:56 +00:00
|
|
|
public function getStandardDeviation(): float
|
2010-08-28 22:05:41 -04:00
|
|
|
{
|
2023-08-02 09:15:01 +00:00
|
|
|
return $this->standardDeviation;
|
2010-08-28 22:05:41 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* A conservative estimate of skill based on the mean and standard deviation.
|
|
|
|
*/
|
2023-08-02 09:04:56 +00:00
|
|
|
public function getConservativeRating(): float
|
2010-08-28 22:05:41 -04:00
|
|
|
{
|
2023-08-02 09:15:01 +00:00
|
|
|
return $this->mean - $this->conservativeStandardDeviationMultiplier * $this->standardDeviation;
|
2010-08-28 22:05:41 -04:00
|
|
|
}
|
|
|
|
|
2023-08-02 09:36:44 +00:00
|
|
|
public function getPartialUpdate(Rating $prior, Rating $fullPosterior, float $updatePercentage): Rating
|
2010-08-28 22:05:41 -04:00
|
|
|
{
|
|
|
|
$priorGaussian = new GaussianDistribution($prior->getMean(), $prior->getStandardDeviation());
|
2023-08-02 09:36:44 +00:00
|
|
|
$posteriorGaussian = new GaussianDistribution($fullPosterior->getMean(), $fullPosterior->getStandardDeviation());
|
2010-08-28 22:05:41 -04:00
|
|
|
|
|
|
|
// From a clarification email from Ralf Herbrich:
|
2022-07-05 15:33:34 +02:00
|
|
|
// "the idea is to compute a linear interpolation between the prior and posterior skills of each player
|
2010-08-28 22:05:41 -04:00
|
|
|
// ... in the canonical space of parameters"
|
|
|
|
|
|
|
|
$precisionDifference = $posteriorGaussian->getPrecision() - $priorGaussian->getPrecision();
|
2016-05-24 14:10:39 +02:00
|
|
|
$partialPrecisionDifference = $updatePercentage * $precisionDifference;
|
2010-08-28 22:05:41 -04:00
|
|
|
|
2023-08-02 09:04:56 +00:00
|
|
|
$precisionMeanDifference = $posteriorGaussian->getPrecisionMean() - $priorGaussian->getPrecisionMean();
|
2016-05-24 14:10:39 +02:00
|
|
|
$partialPrecisionMeanDifference = $updatePercentage * $precisionMeanDifference;
|
2010-08-28 22:05:41 -04:00
|
|
|
|
|
|
|
$partialPosteriorGaussion = GaussianDistribution::fromPrecisionMean(
|
|
|
|
$priorGaussian->getPrecisionMean() + $partialPrecisionMeanDifference,
|
2016-05-24 14:10:39 +02:00
|
|
|
$priorGaussian->getPrecision() + $partialPrecisionDifference
|
|
|
|
);
|
2010-08-28 22:05:41 -04:00
|
|
|
|
2023-08-02 09:15:01 +00:00
|
|
|
return new Rating($partialPosteriorGaussion->getMean(), $partialPosteriorGaussion->getStandardDeviation(), $prior->conservativeStandardDeviationMultiplier);
|
2010-08-28 22:05:41 -04:00
|
|
|
}
|
|
|
|
|
2022-07-05 16:21:06 +02:00
|
|
|
public function __toString(): string
|
2010-08-28 22:05:41 -04:00
|
|
|
{
|
2023-08-02 09:15:01 +00:00
|
|
|
return sprintf('mean=%.4f, standardDeviation=%.4f', $this->mean, $this->standardDeviation);
|
2016-05-24 14:10:39 +02:00
|
|
|
}
|
2022-07-05 15:55:47 +02:00
|
|
|
}
|