mean; } /** * The standard deviation (the spread) of the rating. This is also known as s. */ public function getStandardDeviation(): float { return $this->standardDeviation; } /** * A conservative estimate of skill based on the mean and standard deviation. */ public function getConservativeRating(): float { return $this->mean - $this->conservativeStandardDeviationMultiplier * $this->standardDeviation; } public function getPartialUpdate(Rating $prior, Rating $fullPosterior, $updatePercentage): Rating { $priorGaussian = new GaussianDistribution($prior->getMean(), $prior->getStandardDeviation()); $posteriorGaussian = new GaussianDistribution($fullPosterior->getMean(), $fullPosterior . getStandardDeviation()); // From a clarification email from Ralf Herbrich: // "the idea is to compute a linear interpolation between the prior and posterior skills of each player // ... in the canonical space of parameters" $precisionDifference = $posteriorGaussian->getPrecision() - $priorGaussian->getPrecision(); $partialPrecisionDifference = $updatePercentage * $precisionDifference; $precisionMeanDifference = $posteriorGaussian->getPrecisionMean() - $priorGaussian->getPrecisionMean(); $partialPrecisionMeanDifference = $updatePercentage * $precisionMeanDifference; $partialPosteriorGaussion = GaussianDistribution::fromPrecisionMean( $priorGaussian->getPrecisionMean() + $partialPrecisionMeanDifference, $priorGaussian->getPrecision() + $partialPrecisionDifference ); return new Rating($partialPosteriorGaussion->getMean(), $partialPosteriorGaussion->getStandardDeviation(), $prior->conservativeStandardDeviationMultiplier); } public function __toString(): string { return sprintf('mean=%.4f, standardDeviation=%.4f', $this->mean, $this->standardDeviation); } }