Bayesian vs Limited Fluctuation Experience Analysis

JuliaActuary is an ecosystem of packages that makes Julia the easiest language to get started for actuarial workflows.
modeling
statistics
experience-analysis
bayesian

Alternate title: Why actuaries should cease to care about the number 1082.

Introduction

This notebook briefly discusses two approaches to experience analysis (rate setting). One traditional method, Limited Fluctuation Credibility (LFC), has been around for a very long time although was never intended to be the most accurate predictive method. However, many actuaries still discuss the notion of “full” or “partially” credible indicating a reliance on the LFC concepts.

The Bayesian approach uses explicit assumptions about the statistical relationships and all data given to the model to make inferences. Small datasets lead to greater uncertainty, while larger datasets never reach a point that could be considered “fully credible”, although the posterior density may narrow considerably.

This notebook argues that the Bayesian approach is superior to the LFC heuristics and should be adopted more widely in actuarial practice.

The example will use an auto-claims dataset and will use the first two years of experience to predict the third.

using CSV
using DataFramesMeta
using Distributions
using Turing
using MCMCChains
using DataFrames
using Logging; Logging.disable_logging(Logging.Warn);
using StatsFuns
using StatisticalRethinking
using CairoMakie

Limited Fluctuation Credibility

The Limited Fluctuation Method was so named because it allowed premiums to fluctuate from year to year based on experience, while limiting those fluctuations by giving less than full credibility to premiums based on limited data. In contrast, setting premium rates by giving full credibility to recent experience could be called the Full Fluctuation Method. While every credibility method serves to limit fluctuations, this method acquired its name because it was the first. The Limited Fluctuation Method, also known as Classical Credibility, is the most widely-used credibility method because it can be relatively simple to apply. Outside North America, this method is sometimes referred to as American Credibility.

Quoted from Atkinson, 2019.

Formulas

The Limited Fluctuation Method components: a credibility weight, \[Z\], an Observed Rate, and a Prior Rate.

\[ \text{Credibility-Weighted Rate} = Z × \text{Observed Rate} + (1 – Z) × \text{Prior Rate} \]

With probability equal to \(LC_p\) that the Observed Rate does not differ from the true rate by more than \(LC_r\).

LC_p = 0.9
LC_r  = 0.05
0.05

\[ \text{Claims for full credibility} = \left(\frac{\text{Z-Score}}{\text{ratio}}\right)^{2} \]

LC_full_credibility = round(Int, (quantile(Normal(), 1 - (1 - LC_p) / 2) / LC_r)^2)
1082

Using the inputs above, we have a z-score corresponding to 0.9 of 1.6448536269514717, so: ( 1.6448536269514717 ÷ 0.05)² ≈ 1082. ““”

Atkinson goes on to nicely summarize the square root method which assigns full credibility, \[Z = 1\], when the number of actual claims equals or exceeds the full credibility threshold of $LC_full_credibility claims.

When the number of claims is less than full credibility:

\[ Z = \sqrt{\frac{\text{no. claims in group}}{\text{no. claims full credibility}}} \]

Issues with Limited Fluctuation

  • Ignores available information:
    • Why does our \(Z\) not vary by exposures?
    • No mechanism for a related group to inform the credibility of another
    • Results in a single point estimate with only approximate measure of estimate uncertainty
  • Relies on a number of assumptions/constraints:
    • That aggregated claims can be approximated by a normal distribution
    • The thresholds of $LC_p and $LC_r are arbitrary (e.g. all of the same arguments against p-value thresholds can apply to this approach) ““”

Bayesian Approach

Bayes’ formula of updating the posterior probability conditioned on the observed data:

\[ P(A\mid B) = \frac{P(B \mid A) P(A)}{P(B)} \]

This is well known to most actuaries… but generally not applied frequently in practice!

The issue is that once the probability distributions and data become non-trivial, the formula becomes analytically intractable. Work over the last several decades has set the stage for addressing these situations by developing algorithms that let you sample the Bayesian posterior, even if you can’t analytically say what that is.

A full overview is beyond the scope of this notebook, which is simply to demonstrate that Limited Fluctuation Credibility is of questionable use in practice and that superior tools exist. For references on modern Bayesian statistics, see the end notes.

Note that this is describing a different, more first-principles approach than the the Buhlman Bayesian approach, which attempts to simply relate group experience to population experience. The Bayesian approach described here is much more general and extensible.

The formulation

Contrary to Limited Fluctuation, the Bayesian approach forces one to be explicit about the presumed structure of the probability model. The flexibility of the statistical model allows one to incorporate actuarial judgement in a quantitative way. For example, in this example we assume that the claims experience of each group informs a global (hyperparameter) prior distribution which we could use as a starting point for a new type of observation. More on this once the data is introduced.

Sample Claims Prediction

The data comes from an Allstate auto-claims data via Kaggle. It contains exposure level information about predictor variable and claim amounts for calendar years 2005-2007.

For simplicity, we will focus on the narrow objective of estimating the claims rate at the level of automobile make and model. We will use the years 2005 and 2006 as the training data set and then 2007 to evaluate the predictive model.

The original data is over 13 million rows, we will load an already summarized CSV and split it into train and test sets based on the Calendar_Year.

train,test = let
    pth = download("https://raw.githubusercontent.com/JuliaActuary/Learn/master/data/condensed_auto_claims.csv")
    df = CSV.read(pth,DataFrame,normalizenames=true)
    df.p_observed = df.claims ./ df.n
    

    train = df[df.Calendar_Year .< 2007,:]
    test  = df[df.Calendar_Year .== 2007,:]

    train, test
    
end
(2413×5 DataFrame
  Row  Calendar_Year  Blind_Model  n      claims  p_observed  Int64          String7      Int64  Int64   Float64    
──────┼───────────────────────────────────────────────────────
    1 │          2005  K.78          6132      41  0.00668624
    2 │          2005  Q.22         35141     270  0.00768333
    3 │          2005  AR.41        18719     174  0.00929537
    4 │          2006  AR.41        18868     196  0.010388
    5 │          2005  D.20          4573      27  0.00590422
    6 │          2006  D.20          5446      23  0.00422328
    7 │          2006  AJ.129       50803     342  0.00673189
    8 │          2006  AQ.17        44018     263  0.00597483
    9 │          2005  AQ.17        42684     278  0.00651298
   10 │          2005  BW.3         37903     215  0.00567237
   11 │          2006  BW.3         37752     195  0.00516529
  ⋮   │       ⋮             ⋮         ⋮      ⋮         ⋮
 2404 │          2005  CA.5             1       0  0.0
 2405 │          2005  J.5              1       0  0.0
 2406 │          2006  J.5              1       0  0.0
 2407 │          2005  CB.8             1       0  0.0
 2408 │          2005  AQ.1             1       0  0.0
 2409 │          2005  AJ.96            1       0  0.0
 2410 │          2005  AJ.51            1       0  0.0
 2411 │          2006  AJ.51            1       0  0.0
 2412 │          2005  BQ.6             1       0  0.0
 2413 │          2006  BQ.6             1       0  0.0
                                             2392 rows omitted, 1289×5 DataFrame
  Row  Calendar_Year  Blind_Model  n       claims  p_observed  Int64          String7      Int64   Int64   Float64    
──────┼────────────────────────────────────────────────────────
    1 │          2007  X.45          99368     789  0.00794018
    2 │          2007  Y.29          55783     435  0.00779807
    3 │          2007  P.18           4923      46  0.0093439
    4 │          2007  X.40           5573      22  0.0039476
    5 │          2007  Y.42          25800     168  0.00651163
    6 │          2007  AH.164        12564      88  0.00700414
    7 │          2007  AH.119         4251      26  0.00611621
    8 │          2007  W.3            7680      60  0.0078125
    9 │          2007  BW.107        10897      53  0.00486372
   10 │          2007  BW.79          6938      35  0.00504468
   11 │          2007  AU.58         53424     508  0.00950883
  ⋮   │       ⋮             ⋮         ⋮       ⋮         ⋮
 1280 │          2007  BW.123            1       0  0.0
 1281 │          2007  V.8               1       0  0.0
 1282 │          2007  Z.18              1       0  0.0
 1283 │          2007  X.14              2       0  0.0
 1284 │          2007  AM.7              1       0  0.0
 1285 │          2007  BU.37             1       0  0.0
 1286 │          2007  AE.6              2       0  0.0
 1287 │          2007  CB.6              1       0  0.0
 1288 │          2007  BU.32             1       0  0.0
 1289 │          2007  CA.5              1       0  0.0
                                              1268 rows omitted)

Discussion of Bayesian model

Each group (make and model combination) has an expected claim rate \(μ_i\), which is informed by the global hyperparameter \(μ\) and variance \(\sigma^2\). Then, the observed claims by group are assumed to be distributed according to a Poisson distribution.

A complete overview of modern Bayesian models is beyond the scope of this notebook, but a few key points:

  • We are forced with the Bayesian approach to be explicit about the assumptions (versus all of the implicit assumptions of alternative techniques like LF)
  • We set priors which are the assumed distribution of model parameters before “learning” from the observations. With enough data, it can result in a posterior that the prior was very skeptical of beforehad.
  • With the volume of data in the training set, the priors we select here are not that important. Some comments on why they were chosen though:
    • The rate of claims is linked with a logistic function, which looks like an integral sign of sorts. logistic(0.0) equals 0.5 while very negative inputs approach 0.0 and positive numbers approach 1.0. We do this so that the rate of claims is always constrained in the range \((0,1)\)
    • μ \sim Normal(-2,4) says that we expect the population of auto claims rate to be less than 0.5 (`logistic(-2) ≈ .12) but very wide range of possible values given the wide standard deviation.
    • σ \sim Exponential(0.25) says that we expect the standard devation of an individual group to be positive, but not super wide.
    • μ_i \sim Normal(μ,σ) is the actual prior for each group’s rate of claim and is informed by the above hyperparameters.
  • @model is a Turing.jl macro which enables interpreting the nice ~ syntax, which makes the Julia code look very similar to the traditional mathematical notation.
  • Note the use of broadcasting with the dot syntax (e.g. the . in .~). This tells Julia to vectorize and fuse the computation.
    • data.claims .~ Poisson.(data.n .* logistic.(μ_i)) means “each value in data.claims is a random outcome (.~) distributed accroding to a corresponding Poisson distribution with \lambda =n \times \text{logistic}(\mu_i) where text{logistic}(\mu_i) is the average claims rate for each group.
@model function model_poisson(data)
    # hyperparameter that informs the prior for each group
    μ ~ Normal(-2,4) 
    σ ~ Exponential(0.25)

    # the random variable representing the average claim rate for each group
    # filldist creates a set of random variable without needing to list out each one
    μ_i ~ filldist(Normal(μ,σ),length(unique(data.Blind_Model)))

    # use the poisson appproximation to the binomial claim outcome with a 
    # logisitc link function to keep the probability between 0 and 1
    data.claims .~ Poisson.(data.n .* logistic.(μ_i))
end
model_poisson (generic function with 2 methods)

Here’s what the prior distribution of claims looks like… and peeking ahead to what the posterior for a single group of claims looks like. See how even though our posterior was very wide (favoring small claims rates), it was dominated by the data to create a very narrow posterior average claims rate.

The model is combined with the data and Turing.jl is used to computationally arrive at the posterior distribution for the parameters in the statistical model, \(\mu\), \(\mu_i\), and \(\sigma\).

mp = let
    # combine the different years in the training set 
    condensed = @chain train begin
        groupby(:Blind_Model)
        @combine begin
            :n = sum(:n)
            :claims = sum(:claims)
        end
    end
    model_poisson(condensed);
end
DynamicPPL.Model{typeof(model_poisson), (:data,), (), (), Tuple{DataFrame}, Tuple{}, DynamicPPL.DefaultContext}(model_poisson, (data = 1238×3 DataFrame
  Row  Blind_Model  n       claims  String7      Int64   Int64  
──────┼─────────────────────────────
    1 │ K.78          14058      94
    2 │ Q.22          71401     532
    3 │ AR.41         37587     370
    4 │ D.20          10019      50
    5 │ AJ.129       100824     693
    6 │ AQ.17         86702     541
    7 │ BW.3          75655     410
    8 │ BW.167        42737     294
    9 │ Y.9           92206     755
   10 │ BH.29         15795     162
   11 │ BW.49         27811     156
  ⋮   │      ⋮         ⋮       ⋮
 1229 │ AM.7              2       0
 1230 │ AE.6              3       0
 1231 │ BU.32             2       0
 1232 │ BG.8              1       0
 1233 │ CA.5              2       0
 1234 │ J.5               2       0
 1235 │ AQ.1              1       0
 1236 │ AJ.96             1       0
 1237 │ AJ.51             2       0
 1238 │ BQ.6              2       0
                   1217 rows omitted,), NamedTuple(), DynamicPPL.DefaultContext())

Bayesian Posterior Sampling

Here’s where recent advances in algorithms and computing power make Bayesian analyis possible. We can’t analytically compute the posterior distribution, but we can genererate samples from the posterior such that the frequency of the sampled result appears in proportion to the true posterior density. This uses a technique called Markov Chain Monte Carlo, abbreviated MCMC. There are different algorithms in this family, and we will use one called the No-U-Turn Sampler (NUTS for short).

The result of the sample function is a set of data containing data that is generated in proportion to the posterior distributions and we will use this data to make predictions and understand the distribution of our parameters.

Run the chain:

cp = sample(mp, NUTS(), 500) # this is the line that runs the MCMC sampler

# ╔═╡ 6b5d2186-78ac-48ae-8529-e0dcd51d0b00
let 
    # sample from the priors before learning from any data
    ch_prior = sample(mp,Prior(),1500)
    
    f = Figure()
    ax = Axis(f[1,1],title="Prior distribution of expected claims rate for a group")
    μ = vec(ch_prior["μ_i[1]"].data)
    hist!(ax,logistic.(μ);bins=50)
    ax2 = Axis(f[2,1],title="Posterior distribution of expected claims rate for a group")
    μ = vec(cp["μ_i[1]"].data)
    hist!(ax2,logistic.(μ);bins=50)
    linkxaxes!(ax,ax2)
    f
end

Visualizing the Posterior Density

This is a plot of the posterior density for all of the many, many \(\mu\) parameters in our model. The black line shows the distribution which comes from our hyperparameters μ and σ. In the event of coming across a new group of interest (in our case, a new make of car), this is the prior distribution for the expected claims rate. The model has learned this from the data itself, and serves to regularize predictions.

Utilitiy Function to Plot the Posterior
"""
    dplot(chain,parameter_string)

plot the posterior density for model parameters matching the given string.
"""
function dplot(ch,param_string)
    f = Figure()
    ax = Axis(f[1, 1])

    # plot each group posterior
   for (ind, param) in enumerate(ch.name_map.parameters)
        if contains(string(param),param_string)
            v = vec(getindex(ch, param).data)
            density!(ax,logistic.(v), color = (:red, 0.0),
    strokecolor = (:red,0.3), strokewidth = 1, strokearound = false, label="Individual group posterior")
        end
   end
    
    # Plot hyperparameters
    hyper_mean = mean(getindex(ch, :μ).data)
    hyper_sigma = mean(getindex(ch, :σ).data)
    d = density!(ax,logistic.(rand(Normal(hyper_mean,hyper_sigma),1000)),color = (:black, 0.),
    strokecolor = (:black,0.3), strokewidth = 3, strokearound = false,label="Hyper-parameter distribution")

    
    xlims!(0.0,0.03)
    hideydecorations!(ax)
    axislegend(ax,unique=true)
    f
end

dplot(cp, "μ")

Predictions and Results

Here we compare four predictive models:

  1. Naive, where the predicted rate for the 2007 year is the average of each groups’ for 2005-2006
  2. Limited Fluctuation Overall, where the “prior” in the LFC formula is the overall mean claim rate for 2005-2006
  3. Limited Fluctuation Year-by-Year where the “prior” in the LFC formula is the claim rate for the ith group in 2005 and updated using 2006 claim rates.
  4. Bayesian approach where the predicted claim rate is based on the bayesian model discussed above.

Predictions

Bayesian approach has lower error

Looking at the accumulated absolute error, the bayesian approach has about 16% lower error than the Limited Fluctuation approaches.

Total Actual to Expected

Here, offsetting errors happen to make the naive and LF year-by-year (the latter is a good proxy for the former) such that their total A/E is better than the Bayesian approach.

Given the lower error of the Bayesian approach, one would expect that over multiple years that it would produce more accurate predictions than the alternative methods.

Conclusion

This notebook shows that the Bayesian approach results in claims predictions with less total predictive error than the limited fluctuation method.

Further thoughts

The Bayesian approach could be extended to improve its accuracy even further:

  • Using vehicle make data to add more hierarchical structure to the statistical model. For example, one may observe that Porsches experience crashes at a higher rate than Volvos. LFC cannot embed that sort of overlapping hierarchy into its framework.
  • The Bayesian hyperparameter provides a framework to think about “unseen” make-model combinations

Downsides to the Bayesian approach

  • Computationally intensive. Complex @models can take very long time to run (many hours), compared to relatively quick frequentest methods like maximum likelihood estimation.

Further Reading

If this notebook has piqued your interest in Bayesian techniques, the following books are recommended learning resources (from easiest to most difficult):

  • Statistical Rethinking
  • Bayes Rules!
  • Bayeisan Data Analysis

Appendices

Additional legwork to get the alternate limited fluctuation approach data:

# An alternate approach to LFC where the first year becomes the prior, adjusted by data from the seonc year.
LF2 = let 
    # split dataset by year and recombine
    df2005 = @subset(train,:Calendar_Year .== 2005)
    df2006 = @subset(train,:Calendar_Year .== 2006)
    df = outerjoin(df2005,df2006,on=:Blind_Model,renamecols= "_2005" => "_2006")

    # use 2005 actuals as LFC prior, and the overall mean if model is missing
    μ_2005 = sum(skipmissing(df.claims_2005)) / sum(skipmissing(df.n_2005))
    df.assumed = coalesce.(df.p_observed_2005,μ_2005)
    
    df.LF2_Z = min.(1, .√(coalesce.(df.claims_2006,0.) ./ LC_full_credibility))
    df.LF2_μ = let Z = df.LF2_Z
        # use the 2005 mean if the model not observed in 2006
        Z .* coalesce.(df.p_observed_2006,μ_2005) .+ (1 .- Z) .* df.assumed
    end
    df
end

claims_summary_posterior = let
    # combine the dataset by model
    df = @chain train begin
        groupby(:Blind_Model)
        @combine begin
            :n = sum(:n)
            :claims = sum(:claims)
        end
    end
    pop_μ = sum(df.claims) / sum(df.n)
    df[!,:pop_μ] .= pop_μ

    ## Bayesian prediction
    # get the mean of the posterior estimate for the ith model group
    means = map(1:length(unique(df.Blind_Model))) do i
        d = logistic.(getindex(cp, Symbol("μ_i[$i]")).data)
        (est = mean(d), se=std(d))
    end

    df.p_observed = df.claims ./ df.n
    df.bayes_μ = [x.est for x in means]
    df.bayes_se = [x.se for x in means]

    ## Limited Fluctuation (square root rule)
    # using overall population mean
    # using the square-root rule
    df.LF_Z = min.(1, .√(df.claims ./ LC_full_credibility))
    df.LF_μ = let Z = df.LF_Z
        Z .* (df.p_observed) .+ (1 .- Z) .* pop_μ
    end

    ## Limited Fluctuation 
    # using the first year as the prior, 2nd year as new data
    # using some additional procesing to get the LF2 dataframe, see appendix
    df2005 = @subset(train, :Calendar_Year .== 2005)
    dict2005 = Dict(
        model => rate 
        for (model, rate) in zip(df2005.Blind_Model,df2005.p_observed)
    )
    
    df = leftjoin(df,LF2[:,[:Blind_Model,:LF2_μ]],on=:Blind_Model,)
    
    df = innerjoin(df,test;on=:Blind_Model,renamecols= "_train" => "_test")

    # predictions on the test set using the predictive rates time exposures
    df.pred_bayes = df.n_test .* df.bayes_μ_train
    df.pred_LF = df.n_test .* df.LF_μ_train
    df.pred_LF2 = df.n_test .* df.LF2_μ_train
    df.pred_naive = df.n_test .* df.p_observed_train

    sort!(df,:n_train,rev=true)
end
1224×18 DataFrame
1199 rows omitted
Row Blind_Model n_train claims_train pop_μ_train p_observed_train bayes_μ_train bayes_se_train LF_Z_train LF_μ_train LF2_μ_train Calendar_Year_test n_test claims_test p_observed_test pred_bayes pred_LF pred_LF2 pred_naive
String7 Int64 Int64 Float64 Float64 Float64 Float64 Float64 Float64 Float64? Int64 Int64 Int64 Float64 Float64 Float64 Float64 Float64
1 K.7 382210 3365 0.0072979 0.00880406 0.00878238 0.000168854 1.0 0.00880406 0.00853456 2007 215223 1798 0.00835413 1890.17 1894.84 1836.83 1894.84
2 X.45 192591 1542 0.0072979 0.0080066 0.00800053 0.000199085 1.0 0.0080066 0.00801192 2007 99368 789 0.00794018 794.997 795.6 796.129 795.6
3 AU.14 188702 1687 0.0072979 0.00894002 0.0089118 0.00021057 1.0 0.00894002 0.00874644 2007 114742 1071 0.00933398 1022.56 1025.8 1003.58 1025.8
4 W.16 150304 1064 0.0072979 0.00707899 0.0070739 0.000218869 0.991647 0.00708082 0.00691906 2007 83039 510 0.00614169 587.41 587.984 574.552 587.832
5 AU.11 133445 1188 0.0072979 0.00890254 0.0088575 0.000264302 1.0 0.00890254 0.00878751 2007 70028 642 0.00916776 620.273 623.427 615.372 623.427
6 BO.38 125206 775 0.0072979 0.0061898 0.00619077 0.000223624 0.846325 0.00636009 0.0061673 2007 68966 437 0.00633646 426.952 438.63 425.334 426.886
7 AO.7 109746 1055 0.0072979 0.00961311 0.00956512 0.000305165 0.987444 0.00958404 0.00956317 2007 59534 543 0.00912084 569.45 570.576 569.334 572.307
8 AJ.58 107538 891 0.0072979 0.00828544 0.00826714 0.000252584 0.907455 0.00819405 0.00824136 2007 52176 394 0.00755136 431.346 427.533 430.001 432.301
9 AJ.52 107252 691 0.0072979 0.00644277 0.00643023 0.000263138 0.799145 0.00661453 0.00644592 2007 51952 292 0.00562057 334.063 343.638 334.879 334.715
10 AJ.129 100824 693 0.0072979 0.00687336 0.00684782 0.000263334 0.8003 0.00695814 0.00685673 2007 48920 323 0.00660262 334.995 340.392 335.431 336.245
11 AU.58 98015 970 0.0072979 0.00989644 0.00980722 0.000301882 0.94683 0.00975828 0.00989406 2007 53424 508 0.00950883 523.941 521.326 528.58 528.708
12 X.38 97642 837 0.0072979 0.00857213 0.00851555 0.000300658 0.879527 0.00841862 0.00854678 2007 49682 396 0.00797069 423.069 418.254 424.621 425.881
13 Y.34 92959 734 0.0072979 0.00789595 0.00785963 0.000281347 0.823634 0.00779048 0.00785037 2007 53780 443 0.00823726 422.691 418.972 422.193 424.644
1213 BU.32 2 0 0.0072979 0.0 0.00630026 0.00164119 0.0 0.0072979 0.0 2007 1 0 0.0 0.00630026 0.0072979 0.0 0.0
1214 CA.5 2 0 0.0072979 0.0 0.00636515 0.00160636 0.0 0.0072979 0.0 2007 1 0 0.0 0.00636515 0.0072979 0.0 0.0
1215 D.18 1 0 0.0072979 0.0 0.00630744 0.0015326 0.0 0.0072979 0.00748894 2007 512 2 0.00390625 3.22941 3.73652 3.83434 0.0
1216 Q.7 1 0 0.0072979 0.0 0.0063304 0.00166114 0.0 0.0072979 0.00748894 2007 3 0 0.0 0.0189912 0.0218937 0.0224668 0.0
1217 BW.139 1 0 0.0072979 0.0 0.00648606 0.00170544 0.0 0.0072979 0.00748894 2007 1 0 0.0 0.00648606 0.0072979 0.00748894 0.0
1218 BG.8 1 0 0.0072979 0.0 0.00645148 0.00169365 0.0 0.0072979 0.00748894 2007 28 0 0.0 0.180642 0.204341 0.20969 0.0
1219 R.35 1 0 0.0072979 0.0 0.00639538 0.00160778 0.0 0.0072979 0.00748894 2007 2 0 0.0 0.0127908 0.0145958 0.0149779 0.0
1220 I.6 1 0 0.0072979 0.0 0.00648387 0.00182023 0.0 0.0072979 0.00748894 2007 1 0 0.0 0.00648387 0.0072979 0.00748894 0.0
1221 BM.8 1 0 0.0072979 0.0 0.00642479 0.00180695 0.0 0.0072979 0.00748894 2007 1 0 0.0 0.00642479 0.0072979 0.00748894 0.0
1222 AJ.96 1 0 0.0072979 0.0 0.00635007 0.00161312 0.0 0.0072979 0.0 2007 1 0 0.0 0.00635007 0.0072979 0.0 0.0
1223 AQ.1 1 0 0.0072979 0.0 0.0063907 0.00171822 0.0 0.0072979 0.0 2007 1 0 0.0 0.0063907 0.0072979 0.0 0.0
1224 AJ.118 1 0 0.0072979 0.0 0.00646154 0.00171032 0.0 0.0072979 0.0 2007 1 0 0.0 0.00646154 0.0072979 0.0 0.0
let df = claims_summary_posterior
    f = Figure()
    ax = Axis(f[1,1],title="Cumulative Predictive Residual Error", subtitle="(Lower is better predictive accuracy)",xlabel=L"$i^{th}$ vehicle group",ylabel=L"absolute error")
    lines!(ax,cumsum(abs.(df.pred_LF .- df.claims_test)),label="Limited Fluctuation Overall",color=(:purple,0.5))
    lines!(ax,cumsum(abs.(df.pred_LF2 .- df.claims_test)),label="Limited Fluctuation Year-by-Year",color=(:blue,0.5))
    lines!(ax,cumsum(abs.(df.pred_bayes .- df.claims_test)),label="Bayesian",color=(:red,0.5))
    lines!(ax,cumsum(abs.(df.pred_naive .- df.claims_test)),label="Naive",color=(:grey10,0.5))
    # xlims!(0,40)
    # Legend(f[1,1],[s1,s2],["LFC"halign=:right,valign=:top)
    axislegend(ax,position=:rb)
    f
end

let 
    post = claims_summary_posterior
    X = @chain vcat(train,test) begin
        groupby(:Calendar_Year)
        @combine :claim_rate = sum(:claims) / sum(:n)
    end
    f = Figure()
    ax = Axis(f[1,1],xlabel="year",ylabel="claims rate")
    scatter!(ax,X.Calendar_Year,X.claim_rate, label="data")
    scatter!(ax,[2007],[sum(post.pred_bayes)/sum(post.n_test)], label="Bayes", marker=:hline,markersize=30)
    scatter!(ax,[2007],[sum(post.pred_LF)/sum(post.n_test)], label="LF Overall",marker=:hline,markersize=30)
    scatter!(ax,[2007],[sum(post.pred_LF2)/sum(post.n_test)], label="LF Year-by-Year",marker=:hline,markersize=30)
    scatter!(ax,[2007],[sum(post.pred_naive)/sum(post.n_test)], label="naive",marker=:hline,markersize=30)
    axislegend(ax,position=:rb)
    ylims!(0.005,0.008)
    f
end

Some Remarks about the Results

With limited, real-world data drawing conclusions is a little bit messy because we don’t know what the ground-truth should be, but here are some thoughts that seem to be consistent with what the data and results suggest:

  • The Bayesian approach partially pools the data: it’s an approach in-between assuming each group is independent from the rest and assuming that a single rate applies to all exposures.
  • The partial pooling limits over-fitting, which happens with the naive approach. Our Bayesian approach is somewhat skeptical of groups with low observation counts that stand out from the rest. But it also doesn’t forgoe useful data, as it learns from even low exposures according to what’s consistent with probability theory (Baye’s rule).
  • The naive approach is pretty close to the textbook definition of over-fitting. The LF approaches appear to be under-fitting group-level as there are only $(sum(claims_summary_posterior.LF_Z_train .>= .9999)) groups with “full credibility” (Z=1), even though there are groups with less than “full credibility” with over 100,000 observations in the training set.