分类模型的性能评估——以SAS|分类模型的性能评估——以SAS Logistic回归为例(1): 混淆矩阵
Posted on2008/12/25 by胡江堂 跑完分类模型(Logistic回归、决策树、神经网络等),我们经常面对一大堆模型评估的报表和指标,如Confusion Matrix、ROC、Lift、Gini、K-S之类(这个单子可以列很长),往往让很多在业务中需要解释它们的朋友头大:“这个模型的Lift是4,表明模型运作良好。——啊,怎么还要解释ROC,ROC如何如何,表明模型表现良好……”如果不明白这些评估指标的背后的直觉,就很可能陷入这样的机械解释中,不敢多说一句,就怕哪里说错。本文就试图用一个统一的例子(SAS Logistic回归),从实际应用而不是理论研究的角度,对以上提到的各个评估指标逐一点评,并力图表明:
- 这些评估指标,都是可以用白话(plain English, 普通话)解释清楚的;
- 它们是可以手算出来的,看到各种软件包输出结果,并不是一个无法探究的“黑箱”;
- 它们是相关的。你了解一个,就很容易了解另外一个。
数据 本文使用一个在信用评分领域非常有名的免费数据集,German Credit Dataset,你可以在UCI Machine Learning Repository找到(下载;数据描述)。另外,你还可以在SAS系统的Enterprise Miner的演示数据集中找到该数据的一个版本(dmagecr.sas7bdat)。以下把这个数据分为两部分,训练数据train和验证数据valid,所有的评估指标都是在valid数据中计算(纯粹为了演示评估指标,在train数据里计算也未尝不可),我们感兴趣的二分变量是good_bad,取值为{good, bad}:
Train datagood_badFrequencyPercent-------------------------------------------bad15425.67good44674.33Valid datagood_badFrequencyPercent--------------------------------------------bad14636.50good25463.50
信用评分指帮助贷款机构发放消费信贷的一整套决策模型及其支持技术。一般地,信用评分技术将客户分为好客户与坏客户两类,比如说,好客户(good)能够按期还本付息(履约),违约的就是坏客户(bad)。具体做法是根据历史上每个类别(履约、违约)的若干样本,从已知的数据中考察借款人的哪些特征对其拖欠或违约行为有影响,从而测量借款人的违约风险,为信贷决策提供依据。Logistic回归是信用评分领域运用最成熟最广泛的统计技术。
约定 在我们的示例数据中,要考察的二分变量是good_bad,我们把感兴趣的那个取值bad(我们想计算违约的概率),称作正例(Positive, 1),另外那个取值(good)称作负例(Negative, 0)。在SAS的Logistic回归中,默认按二分类取值的升序排列取第一个为positive,所以默认的就是求bad的概率。(若需要求good的概率,需要特别指定)。
模型 如果没有特别说明,以下所有的SAS代码都在SAS 9.1.3 SP4系统中调试并运行成功(在生成ROC曲线时,我还会提到SAS9.2的新功能)。
proc logistic data=https://www.it610.com/article/train; model good_bad=checking history duration savings property; run;
这个数据很整齐,能做出很漂亮的模型,以下就直接贴出参数估计的结果:
Analysis of Maximum Likelihood EstimatesStandardWaldParameterDFEstimateErrorChi-SquarePr > ChiSqIntercept10.60320.44661.82420.1768checking1-0.65360.093149.3333<.0001history1-0.40830.098017.3597<.0001duration10.02480.009077.48200.0062savings1-0.20170.07457.33080.0068property10.31570.10529.01630.0027
【分类模型的性能评估——以SAS|分类模型的性能评估——以SAS Logistic回归为例(1): 混淆矩阵】回归方程就是:
logit[p(bad)]=log(p/1-p)=0.6032-0.6536*checking-0.4083*history+0.0248*duration-0.2017*savings+0.3157*property
用下面的公式就可以求出正例的概率(bad的概率):
p=exp(logit)/(exp(logit)+1)
上式求出的是概率值,如何根据概率值把各个客户归类,还需要一个阈值,比如,这里我们简单地规定,违约概率超过0.5的就归为bad,其余为good。把上述公式代入valid数据中,
data valid_p; set valid; logit=0.6032-0.6536*checking-0.4083*history+0.0248*duration-0.2017*savings+0.3157*property; p=exp(logit)/(exp(logit)+1); if p<0.5 then good_bad_predicted='good'; else good_bad_predicted='bad'; keep good_bad p good_bad_predicted; run;
从下面的局部的数据valid_p可以看到,一些实际上是good的客户,根据我们的模型(阈值p取0.5),却预测他为bad(套用我们假设检验的黑话,这就犯了“弃真”的错误),对一些原本是bad的客户,却预测他为good(“取伪”错误),当然,对更多的客户,good还预测成good,bad还预测成bad:
good_badpgood_bad_predictedbad0.61624badbad0.03607goodgood0.12437goodgood0.21680goodgood0.34833goodgood0.69602badbad0.68873badgood0.48351goodgood0.03288goodgood0.06789goodgood0.61195badgood0.15306good
Confusion Matrix, 混淆矩阵 一个完美的分类模型就是,如果一个客户实际上(Actual)属于类别good,也预测成(Predicted)good,处于类别bad,也就预测成bad。但从上面我们看到,一些实际上是good的客户,根据我们的模型,却预测他为bad,对一些原本是bad的客户,却预测他为good。我们需要知道,这个模型到底预测对了多少,预测错了多少,混淆矩阵就把所有这些信息,都归到一个表里:
预测1 0实 1 d, True Positive c, False Negative c+d, Actual Positive际 0 b, False Positive a, True Negative a+b, Actual Negativeb+d, Predicted Positive a+c, Predicted Negative
其中,
- a是正确预测到的负例的数量, True Negative(TN,0->0)
- b是把负例预测成正例的数量, False Positive(FP, 0->1)
- c是把正例预测成负例的数量, False Negative(FN, 1->0)
- d是正确预测到的正例的数量, True Positive(TP, 1->1)
- a+b是实际上负例的数量,Actual Negative
- c+d是实际上正例的个数,Actual Positive
- a+c是预测的负例个数,Predicted Negative
- b+d是预测的正例个数,Predicted Positive
同样,犯错也有两种情况。实际是Positive,预测成Negative (1->0) ,这就是False Negative;实际是Negative,预测成Positive (0->1) ,这就是False Positive;
我们可以通过SAS的proc freq得到以上数字:
proc freq data=https://www.it610.com/article/valid_p; tables good_bad*good_bad_predicted/nopercent nocol norow; run;
对照上表,结果如下:
预测1 0实 1,bad d, True Positive,48 c, False Negative,98 c+d, Actual Positive,146际 0,good b, False Positive,25 a, True Negative,229 a+b, Actual Negative,254b+d, Predicted Positive,73 a+c, Predicted Negative,327 400
根据上表,以下就有几组常用的评估指标(每个指标分中英文两行):
1. 准确(分类)率VS.误分类率
准确(分类)率=正确预测的正反例数/总数
Accuracy=true positive and true negative/total cases= a+d/a+b+c+d=(48+229)/(48+98+25+229)=69.25%
误分类率=错误预测的正反例数/总数
Error rate=false positive and false negative/total cases=b+c/a+b+c+d=1-Accuracy=30.75%
2. (正例的)覆盖率VS. (正例的)命中率
覆盖率=正确预测到的正例数/实际正例总数,
Recall(True Positive Rate,or Sensitivity)=true positive/total actual positive=d/c+d=48/(48+98)=32.88%
/*注:覆盖率(Recall)这个词比较直观,在数据挖掘领域常用。因为感兴趣的是正例(positive),比如在信用卡欺诈建模中,我们感兴趣的是有高欺诈倾向的客户,那么我们最高兴看到的就是,用模型正确预测出来的欺诈客户(True Positive)cover到了大多数的实际上的欺诈客户,覆盖率,自然就是一个非常重要的指标。这个覆盖率又称Sensitivity, 这是生物统计学里的标准词汇,SAS系统也接受了(谁有直观解释?)。 以后提到这个概念,就表示为, Sensitivity(覆盖率,True Positive Rate)。 */
命中率=正确预测到的正例数/预测正例总数
Precision(Positive Predicted Value,PV+)=true positive/ total predicted positive=d/b+d=48/(48+25)=65.75%
/*注:这是一个跟覆盖率相对应的指标。对所有的客户,你的模型预测,有b+d个正例,其实只有其中的d个才击中了目标(命中率)。在数据库营销里,你预测到b+d个客户是正例,就给他们邮寄传单发邮件,但只有其中d个会给你反馈(这d个客户才是真正会响应的正例),这样,命中率就是一个非常有价值的指标。 以后提到这个概念,就表示为PV+(命中率,Positive Predicted Value)*。/
3.SPECIFICITY VS. PV-
负例的覆盖率=正确预测到的负例个数/实际负例总数
Specificity(True Negative Rate)=true negative/total actual negative=a/a+b=229/(25+229)=90.16%
/*注:Specificity跟Sensitivity(覆盖率,True Positive Rate)类似,或者可以称为“负例的覆盖率”,也是生物统计用语。以后提到这个概念,就表示为Specificity(负例的覆盖率,True Negative Rate) 。*/
负例的命中率=正确预测到的负例个数/预测负例总数
Negative predicted value(PV-)=true negative/total predicted negative=a/a+c=229/(98+229)=70.03%
/*注:PV-跟PV+(命中率,Positive Predicted value)类似,或者可以称为“负例的命中率”。 以后提到这个概念,就表示为PV-(负例的命中率,Negative Predicted Value)。*/
以上6个指标,可以方便地由上面的提到的proc freq得到:
proc freq data=https://www.it610.com/article/valid_p; tables good_bad*good_bad_predicted ; run;
其中,准确率=12.00%+57.25%=69.25% ,覆盖率=32.88% ,命中率=65.75% ,Specificity=90.16%,PV-=70.03% 。
或者,我们可以通过SAS logistic回归的打分程序(score)得到一系列的Sensitivity和Specificity,
proc logistic data=https://www.it610.com/article/train; model good_bad=checking history duration savings property; score data=valid outroc=valid_roc; run;
数据valid_roc中有几个我们感兴趣的变量:
- _PROB_:阈值,比如以上我们选定的0.5
- _SENSIT_:sensitivity(覆盖率,true positive rate)
- _1MSPEC_ :1-Specificity,为什么提供1-Specificity而不是Specificity,下文有讲究。
_PROB_ _SENSIT_ _1MSPEC_0.54866 0.26712 0.070870.54390 0.27397 0.078740.53939 0.28767 0.086610.52937 0.30137 0.090550.51633 0.31507 0.094490.50583 0.32877 0.098430.48368 0.36301 0.102360.47445 0.36986 0.10630
如果阈值选定为0.50583,sensitivity(覆盖率,true positive rate)就为0.32877,Specificity就是1-0.098425=0.901575,与以上我们通过列联表计算出来的差不多(阈值0.5)。
下期预告:ROC 以上我们用列联表求覆盖率等指标,需要指定一个阈值(threshold)。同样,我们在valid_roc数据中,看到针对不同的阈值,而产生的相应的覆盖率。我们还可以看到,随着阈值的减小(更多的客户就会被归为正例),sensitivity和1-Specificity也相应增加(也即Specificity相应减少)。把基于不同的阈值而产生的一系列sensitivity和Specificity描绘到直角坐标上,就能更清楚地看到它们的对应关系。由于sensitivity和Specificity的方向刚好相反,我们把sensitivity和1-Specificity描绘到同一个图中,它们的对应关系,就是传说中的ROC曲线,全称是receiver operating characteristic curve,中文叫“接受者操作特性曲线”。欲知后事如何,且听下回分解。
参考资料:
- Mithat Gonen. 2007. Analyzing Receiver Operating Characteristic Curves with SAS. Cary, NC: SAS Institute Inc.
- Dan Kelly, etc. 2007. Predictive Modeling Using Logistic Regression Course Notes. Cary, NC: SAS Institute Inc.
- Confusion Matrix, seehttp://www2.cs.uregina.ca/~dbd/cs831/notes/confusion_matrix/confusion_matrix.html
推荐阅读
- 热闹中的孤独
- JAVA(抽象类与接口的区别&重载与重写&内存泄漏)
- 放屁有这三个特征的,请注意啦!这说明你的身体毒素太多
- 一个人的旅行,三亚
- 布丽吉特,人生绝对的赢家
- 慢慢的美丽
- 尽力
- 一个小故事,我的思考。
- 家乡的那条小河
- 《真与假的困惑》???|《真与假的困惑》??? ——致良知是一种伟大的力量