The previous part introduced the Bayes decision procedure to find the optimal threshold, i.e. the cut-off score that transforms a recognizer into a classifier that achieves the minimum expected risk.

The Bayes rules simply compared likelihood ratios with an application-dependent ratio to assign a class. The Receiving Operator Characteristics (ROC) has a focus on error trade-offs rather than likelihood ratios, which allows to not only find optimal thresholds but also to run and visualise various investigations, all on one graph.

The first section looks at threshold optimisation using the ROC curve, and shows why it is equivalent to using the Bayes decision procedure. Understanding the connections between different evaluation frameworks is essential to compare and choose the best approach. The rest covers various topics related to ROC analysis:

  • Section 2: why ROC convexity matters and how to enforce it with the PAV algorithm
  • Section 3: the connection between ROC and concordance probability via AUC, and why you may not want to optimise for the latter
  • Section 4: benchmarking a classifier with majority rules using ROC analysis

A. The ROC curve

Finding an optimal threshold

The ROC curve slides through every possible cut-off point in descending order and plots the corresponding (Pfa, 1−Pmiss) values at that cut-off. Pmiss and Pfa are error rates estimated as proportions of errors of class $\omega_1$ and $\omega_0$, resp. (1-Pmiss) is often called the true positive rate (tpr), because Pmiss is estimated as the proportion of targets below the cut-off point, which is also the rate of false negatives. Similarly, Pfa is the proportion of non-targets labeled as false positives, also called false positive rate (fpr). And so, (Pfa, 1-Pmiss) can be referred to as (fpr, tpr), or just an “operating point”.

source

  case class Tradeoff(...) {
        // ...

    val asROC: Matrix = {
      // rhsArea (right-hand side area) is 1 - cumulative distribution function
      val fpr = (rhsArea andThen decreasing)(w0Counts)
      val tpr = (rhsArea andThen decreasing)(w1Counts)
      Vector(fpr, tpr)
    }
  }

Combining error rates with the application-dependent priors and error costs yields expected risks. In the ROC space, isocost lines plot operating points that correspond to the same expected risk. So, the intersection between a ROC operating point and the isocost line tells us what risk to expect at that cut-off point. Doing this for every ROC operating point and “argmin-ing” returns the optimal threshold.

The bottom graph below plots the ROC curve with the isocost line - red-dotted line with slope 0.2 - for the fraud application of Part 1, AppParameters(p_w1=0.5,Cmiss=25,Cfa=5). The isocost intersects the ROC curve at the optimal operating point.

Every other parallel line would also be an isocost, but it would intersect at a sub-optimal point because either (1-Pmiss) would be lower and/or Pfa would be higher. That means that the optimal operating point is where the isocost line is tangent to the ROC.

Bayes Decisions with the ROC

Connection to the Bayes decision procedure

As the CCD graph above suggets, the ROC analysis is connected to the Bayes decision rule. As a reminder, the rule instructs to choose targets when the cost-weighted prior ratio, \(\delta = \frac{p(\omega_0) \times \text{Cfa}}{p(\omega_1) \times \text{Cmiss}}\), is equal to the likelihood ratio.

Going back to the observation that “the optimal threshold is where the isocost line is tangent to the ROC”, the key is that ROC operating points correspond to likelihood-ratios, and isocost lines have a gradient of $\delta$, so ROC optimization is equivalent to a Bayes decision.

Isocost lines have a derivative equal to $\delta$

Starting with equation 1.2, we get the isocost defined as a linear relationship between (1-Pmiss) and Pfa with $\delta$ as the derivative

\[\begin{equation} \begin{aligned} & p(\omega_1)*\text{Pmiss}*\text{Cmiss} p(\omega_1)*\text{Pfa}*\text{Cfa}=\text{e} \\ & \text{Pmiss} = -\text{Pfa}*\frac{p(\omega_0*Cfa)}{p(\omega_1)*\text{Cmiss}} \frac{\text{e}}{p(\omega_1)*\text{Cmiss}} \\ & 1-\text{Pmiss} = \text{Pfa}*e^{-\theta} (1-\frac{\text{e}}{p(\omega_1)*\text{Cmiss}}) \end{aligned} \end{equation}\]

And so, \(\text{tpr} = \text{fpr}*\delta \text{a} \tag{2.1}\)

with $\delta = \frac{p(\omega_0) \times \text{Cfa}}{p(\omega_1) \times \text{Cmiss}}$ and $\text{a}=\frac{p(\omega_1) \times \text{Cmiss} - E(\text{risk})}{p(\omega_1) \times \text{Cmiss}}$.

And $\delta$ is the right-hand side in equation 1.1

ROC curve gradients are likelihood ratios

First, the derivative of the ratio wrt scores, is the ratio of the derivatives: ($\frac{\text{1-Pmiss}}{\text{Pfa}})’ = \frac{\text{(1-Pmiss)’}}{\text{(Pfa)’}}$. Then, Pmiss (Pfa) is based on the cumulative distribution function of scores conditionned on $\omega_1$ ($\omega_0$), so the derivative of Pmiss (Pfa) is just the probability of targets (non-targets), and the ROC gradient is the likelihood ratio $\frac{p(s \vert \omega_1)}{p(s \vert \omega_0)}$.

Let’s double-check with the numerical example below, which calculates two values for every operating point: lr gets the likelihood ratio directly from the Tradeoff object’s counts, while slope gets the lr from the slope of (tpr,fpr). Both values are the same, so the derivative of the ROC is the likelihood ratio.

Note that the ROC curve is not a continuous function but a line, so its slope is the segment that connects a point to the next one. The slope at $(1-\text{Pmiss(t)},\text{Pfa(t)})$ is $\frac{ \text{Pmiss(t)}-\text{Pmiss(t-1)} }{ \text{Pfa(t-1)}-\text{Pfa(t)} }$.

source

      /* lr(w1) =
                    p(s|w1)/p(s|w0)
       */
      def lr(to: Tradeoff) =
        pdf(to.w1Counts).zip(pdf(to.w0Counts)).map(tup => tup._1 / tup._2)

      /* slope =
                    pmiss(t)-pmiss(t-1) / pfa(t-1)-pfa(t)
       */
      def slope(to: Tradeoff) = {
        val pMissD = to
          .asPmissPfa(0)
          .sliding(2)
          .map { case Seq(x, y, _*) => y - x }
          .toVector
        val pFaD = to
          .asPmissPfa(1)
          .sliding(2)
          .map { case Seq(x, y, _*) => x - y }
          .toVector
        pMissD.zip(pFaD).map(tup => tup._1 / tup._2)
      }

And lr and slopereturn the same results as expected.

@ lr(hisTo)
res1: Vector[Double] = Vector(0.0, 0.0, 0.0, 0.0, 0.007871325628334975, 0.0, 0.0, 0.01250151717441437, 0.07484429961857492, 0.19986233803966602, 0.6509123275478416, 1.9127321276853988, 5.6291370950741335, 20.78029965880433, 55.46923170287657, 82.21206052514464, 360.01869158878503, Infinity, Infinity, Infinity, Infinity, Infinity, NaN, Infinity, NaN)
@ slope(hisTo)
res2: Vector[Double] = Vector(0.0, 0.0, 0.0, 0.0, 0.007871325628334973, 0.0, 0.0, 0.012501517174414365, 0.07484429961857492, 0.19986233803966608, 0.6509123275478419, 1.912732127685398, 5.629137095074125, 20.780299658804417, 55.4692317028767, 82.21206052514438, 360.01869158876053, Infinity, Infinity, Infinity, Infinity, Infinity, NaN, Infinity, NaN)

Benefits of a ROC-based analysis

The ROC curve emphasizes the tradeoff between missing and false alarm errors in determining the optimal decision. Starting in the bottom left corner where $\text{Pmiss}=1$ and $\text{Pfa}=0$, and moving up the curve by decreasing the threshold, we get a lower miss rate if we accept a higher false alarm rate as long as the curve is not flat. How much $\text{Pfa}$ we tolerate is determined by the application type, captured by the derivative of the isocost line. Given flat priors and high Cmiss, our application requires low $\text{Pmiss}$ and thus tolerates high Pfa. Visually, that corresponds to operating points in the right upper-hand corner.

If the application type had a more balanced ratio of cost-weighted priors, the isocost slope would be approximately parallel to the $tpr=fpr$ line, e.g. when minimising error rates with equal prior probabilities (Cmiss = Cfa = 1 and $p(\omega_1)=p(\omega_0)=0.5$). In that case, we would be looking at operating points near the middle bump of the curve. This type of visual inspection is even more useful when comparing multiple recognizers, as a quick glance indicates whose isocost is closer to the top left corner, i.e. which is the better performer1.

B. The PAV algorithm

ROC-based threshold optimisation loops through every score to compute its tangent, with the result being the closest to $\delta$. As a side note, popular implementations like R’s ROCR or python’s sklearn calculate the expected risk instead of the tangent, then find the minimum. At any rate, looping through all scores requires being careful about how we construct intervals to balance accuracy vs efficiency - large bins risk missing the true optimal threshold, whereas thin bins include many unsuitable candidates that could waste resources.

For example, Tradeoff starts with a histogram of score counts, but the issue is then to define bin width. If it’s too wide, the optimal cut-off may get lost inside a bin, resulting in a higher expected risk than we would have with thinner bins. Conversely, if the bins are too thin, some points will not be reliable - a usual problem when estimating continuous distributions from samples.

Below, the blue (LLR) and green (ROC) estimates are built on very thin intervals. The blue curve goes up and down, i.e. it’s not a monotonous function of scores, and the green ROC curve has a steppy pattern, i.e. it’s not convex. Not only is the result visually unpleasant, it is also inefficient because it is possible to construct a threshold with lower fpr and/or higher tpr than any point below the ROC convex hull.

It would be good to exclude non-optimal thresholds to avoid wasting compute resources2. That can be achieved with the PAV algorithm, which guarantees non-decreasing LLR and convex ROC curves, as the orange (LLR) and red (ROC) curves show. For more information and some compelling examples, I would refer to T. Fawcett and A. Niculescu-Mizil (2007).

LLR and ROC curves - Histogram vs PAV

C. Risk VS AUC use case

ROC curves offer it possible to visually inspect the concordance probability covered in Part 1, because the Area Under the Curve (AUC) of the ROC is equal to $p(s_{\omega_0} < s_{\omega_0})$ - see this SO question for a proof. What is not clear so far is why we should care about concordance probabilities. In particular, should we aim for a minimum AUC?

The next example shows that higher concordance is not necessarily better from a risk optimisation viewpoint. Hence, if our objective is to minimise risk, we need not pay too much attention to AUC, which may at best be a good sense check - e.g. is it close or well above 0.5?

The example is based on ML Meets Economics, which is a great practical introduction to AUC and ROC curves. I have made minor adjustments to their numbers to fit my simulated data.

Imagine a factory making widgets with the following data

  • Widgets overheat 5% of the time due to faulty gearboxes
  • Bad gearboxes cost the company £157 owing to wasted labour and inputs
  • The gearbox supplier, which sells them at £50 apiece, won’t improve quality controls, so the factory decides to use an ML system to predict which gearboxes will overheat
  • A gearbox flagged as faulty must be tested, which destroys it, resulting in a loss equivalent to the item’s cost (£50)
  • A working widget generates a net profit of £40

If $\omega_1$ ($\omega_0$) represents the defect (working) gearbox and $\alpha_1$ ($\alpha_0$) the decision to test (not test) a gearbox, then the cost & profit structure is

graph LR; id0([start]) --> id11[class ω0]; id0([start]) --> id12[class ω1]; id11[class ω0]-->id21[α0]; id11[class ω0]-->id22[α1]; id12[class ω1]-->id31[α0]; id12[class ω1]-->id32[α1]; id21[action α0] --> id41[cost c_00 = -40.0]; id22[action α1] --> id42[cost c_10 = Cfa = 50.0]; id31[action α0] --> id43[cost c_01 = Cmiss = 157.0]; id32[action α1] --> id44[cost c_11 = 50.0];

Note the negative cost of not testing the item when it’s not faulty, which corresponds to a net profit. In the original blog article, the author maximises utility, while we minimise the expected risk, but the two approaches are equivalent (as long as we use signs consistently).

We can use the simplified cost structure implemented in AppParameters by identifying the extra terms and offsetting them. This is actually a useful exercise to realise that the Bayes decision threshold $-\theta$ is only determined by the ratio between \(p(\omega_0)*u\) and \(p(\omega_1)*v\) . We decided that $u$ and $v$ should be Cmiss and Cfa, resp., but we could break them down differently.

The Bayes decision criterion is to choose $\alpha_1$ if $\text{lr}(\omega_1)$ is greater than $\frac{(\text{C10-C00})p(\omega_0)}{(\text{C01-C11})p(\omega_1)} = \frac{90 \times p(\omega_0)}{107 \times p(\omega_1)}$, and so the cost structure is equivalent to AppParameters(p_w1=0.05,Cmiss=107,Cfa=90).

In what follows, we evaluate the high- and low-AUC recognizers, which are prefixed with “hi” and “low”, resp. I use the probability_monad framework to craft and sample from the score distributions — defined here.

We start by checking that the AUC of the hiAUCdata recognizer (highAUC) is truly higher using the smartA function described in Part 1. In probability_monad, the pr method samples from the rv to evaluate a predicate. Here, we check that the difference in AUC is positive, i.e. that hiAUCdata’s AUC is bigger than lowAUCdata.

source

    def hiAUCdata: Distribution[List[Score]] = HighAUC.normalLLR.repeat(1000)
    def lowAUCdata: Distribution[List[Score]] = LowAUC.normalLLR.repeat(1000)

    def splitScores(data: List[Score]): Tuple2[Row, Row] = {
      val tarS = data.filter(_.label == 1).map(_.s).toVector
      val nonS = data.filter(_.label == 0).map(_.s).toVector
      (nonS, tarS)
    }

    def score2Auc(data: List[Score]): Double = {
      val (nonS, tarS) = splitScores(data)
      smartA(nonS, tarS)
    }

    def simAUC: Distribution[Double] = for {
      h <- hiAUCdata
      l <- lowAUCdata
      hAuc = score2Auc(h)
      lAuc = score2Auc(l)
      diff = (hAuc - lAuc)
    } yield diff

    /* Check that
        P(hiAuc > lowAuc) 
    */
    val p = simAUC.pr(_ > 0)
    val α = 0.05
@ println(p)
res3: Double = 0.9775823313628146
@ assert(p > (1-α))
res4: Boolean = true

The hiAUCdatarecognizer is better from an AUC viewpoint, however, it has a worse expected risk on the sample dataset:

    val aucPa = AppParameters(0.05,107,90)

    // Define w1HiCnts etc. using PAV - see source code
    val hiTo = Tradeoff(w1HiCnts,w0HiCnts,hiThresh)
    val lowTo = Tradeoff(w1LowCnts,w0LowCnts,lowThresh)
@ println(hiTo.minRisk(aucPa))
res3: Double = 5.16
@ println(lowTo.minRisk(aucPa))
res4: Double = 2.81

The LLR and ROC plots help to make sense of the disagreement between AUC and $E(\text{risk})$. First, the LLR plot confirms that hiAUC’s AUC is higher because its LLR line is steeper. lowAUC’s LLR line is flatter around 0 for negative scores, and by definition, LLR values near 0 means have no discrimination power.

The application’s high $-\theta$ means that false positives are penalised more than false negatives, so the optimal cut-off point will be in the bottom-left corner, where fpr remains high. The better performing recognizer will have the steeper ROC curve, and that’s not highAUC. Starting from the bottom-left corner at (Pmiss=1, Pfa=0), highAUC’s derivative quickly drops below the isocost derivative, which leaves the optimal threshold close to the maximum observed score. That makes hiAUC not very different from a dummy rule that assigns every instance to non-target. We shall call this extreme case an all-$\omega_0$ classifier.

lowAUC’s ROC curve is very steep in the high Pmiss / low Pfa region, which makes it possible for lowAUC to identify more positives than hiAUC while keeping fpr low enough, which translates into a lower expected risk. Past the 0.1 mark, lowAUC quickly flattens, which would make its performance worse for most application types, hence its lower AUC. The key point is that AUC measures the overall discrimination ability, i.e. across all application types, but it doesn’t say how a classifier performs on specific application types.

High AUC (Left) vs Low AUC (Right)

D. Majority-isocost lines

The all-$\omega_0$ “classifier” mentioned above is a simple rule that assigns all instances to non-target. Another name may be majority classifier because it assigns the class with the highest cost-weighted prior probability3. Its expected risk is a benchmark that any recognizer should beat because it is inexpensive, simple, and thus it is often the status quo before any automated solution.

In the previous example, the all-$\omega_0$ expected risk is $p(\omega_1) \times \text{Pmiss} \times \text{Cmiss} = 0.05 \times 1 \times £107 = £5.35$, i.e. close to hiAUC’s risk of £5.16, which thus does not add a lot of value. If hiAUC was our only option, it would be a good idea to double-check its costs to ensure that it’s better than the simple rule.

It is useful to add the isocost line of the majority rule - let’s call it the majority-isocost line - as a benchmark, as it indicates if using a recognizer makes any sense. If a recognizer’s ROC curve is below the the majority-isocost line, its expected risk under any application type would be higher than the majority rule, so it does not make economic sense to use it. Hence, the majority-isocost line can inform go/no-go decisions about building ML solutions.

Besides, this benchmark is useful for sensitivity analyses of different application scenarios. Deciding on the application type is somewhat of a subjective exercise, so using a range of inputs is preferable if it’s possible. For example, an increase in supplier costs, a change in regulation impacting labour wage or an improvement in the supplier’s defect rates all would impact the operating context, and therefore the expected risk of a recognizer. Visualising these scenarios via the majority-isocost lines can provide reassurance that a recognizer adds value even in the worst case.

The graph below shows the previous parameters and another scenario that penalises false positives more, which corresponds to the steeper line. It’s interesting that lowAUC still has points above the steeper isocost.

High AUC (Left) vs Low AUC (Right)

A note on the calculation - the majority-isocost line is defined as $\text{(tpr-d)} = \text{(fpr-d)} \times \frac{p(\omega_0) \times \text{Cfa}}{p(\omega_1) \times \text{Cmiss}}$, with $d$ being (0,0) for the all-$\omega_0$ rule, or (1,1) for the all-$\omega_1$ rule. See the details in the appendix.

To determine which all-$\omega_i$ is the reference, compute their expected-risk ratio

\[\text{rr}=\frac{E(risk_{all-\omega_1})}{E(risk_{all-\omega_0})} = \frac{p(\omega_0) \times \text{Cfa}}{p(\omega_1) \times \text{Cmiss}}\]

And so, the line equation can also be defined as $(\text{tpr}-d) = (\text{fpr}-d) \times \text{rr}$, which has a nice visual interpretation. If the all-non-target rule has the lower risk, the line equation is $\text{tpr} = \text{fpr} \times \text{rr}$ which lies above the $\text{tpr}=\text{fpr}$ line. Otherwise, it is $\text{(tpr-1)} = \text{(fpr-1)} \times \text{rr}$ which also lies above $\text{tpr}=\text{fpr}$.

Furthermore, the majority-isocost is $(\text{tpr}=\text{fpr})$ when $\text{eer} = 1$ i.e. when application parameters don’t carry any information for the Bayes decision threshold. In that case, there is no majority classifier, and we may as well flip a coin to decide which label to choose for all instances.

A recognizer’s ROC is on this line if it can’t do better than this randomly assigned majority rule, which means that this recognizer has a concordance probability (AUC) of 0 - that is often how people refer to $(\text{tpr}=\text{fpr})$.

E. Conclusion

ROC analysis teaches us that finding an optimal threshold is about trading off one error for another. We start with a very high Pmiss, and we give it away in exchange for more Pfa ; the application type determines when we get the best deal, in which case we stop the trade and find the corresponding threshold.

Up until now, scores had no meaning other than higher values corresponding to higher target probability. The next part addresses decision-making using calibrated scores, which can be a better alternative to using raw scores.

Appendix

Details of the majority-isocot equation

Rearrange the expected risk equation (2.1)

\(\text{tpr}=\text{fpr}*\delta \frac{p(\omega_1) \times \text{Cmiss} - E(\text{risk})}{p(\omega_1) \times \text{Cmiss}}\) With $\delta = \frac{p(\omega_0) \times \text{Cfa}}{p(\omega_1) \times \text{Cmiss}}$

At any point (d,d) on $\text{tpr}=\text{fpr}$, \(E(\text{risk}) = p(\omega_0)\times\text{Cfa}\times\text{d} p(\omega_1)\times\text{Cmiss}-p(\omega_1)\times\text{Cmiss}\times\text{d}\)

Plug the $(d,d)$ risk value into the expected risk equation, rearrange to get

\[\text{(tpr-d)} = \text{(fpr-d)} \times \delta\]

References

  • T. Fawcett and F. Provost (2001). Robust Classification for Imprecise Environments.
  • T. Fawcett and A. Niculescu-Mizil (2007). PAV And the ROC Convex Hull.
  1. The ROC framework makes it easy to not only compare but also combine recognizers, which can be a better option when one recognizer achieves better risk only on some parts of the ROC curve. See T. Fawcett and F. Provost (2001) for details and examples. 

  2. Implementations typically use all scores available as thresholds, which can result in a huge number of possible cut-off points - see “Algorithm 2” in T. Fawcett and A. Niculescu-Mizil (2007). In a sense, using scores is similar to constructing histograms with varying-size bin sizes, which gets the lowest bin width such that there is at least a positive value in tpr or fpr. For example, see ROCR .compute.unnormalized.roc.curve

  3. For example, ApplicationParameters(p_w1=0.45, Cmiss=2, Cfa=1) corresponds to an all-$\omega_1$ rule because $p(\omega_1) \times \text{Cmiss}$ > $p(\omega_0) \times \text{Cfa}$