如何在 Python 中构建提升图(a.k.a 增益图)?

How to build a lift chart (a.k.a gains chart) in Python?

我刚刚使用 scikit-learn 创建了一个模型,该模型可以估计客户响应某个提议的可能性。现在我正在尝试评估我的模型。为此,我想绘制提升图。我理解提升的概念,但我很难理解如何在 python.

中实际实现它

Lift/cumulative 增益图表不是评估模型的好方法(因为它不能用于模型之间的比较),而是一种在资源有限的情况下评估结果的方法。要么因为对每个结果采取行动都需要成本(在营销场景中),要么你想忽略一定数量的有保证的选民,只对那些持观望态度的人采取行动。如果您的模型非常好,并且对所有结果都具有很高的分类准确度,那么按信心排序您的结果不会有太大提升。

import sklearn.metrics
import pandas as pd

def calc_cumulative_gains(df: pd.DataFrame, actual_col: str, predicted_col:str, probability_col:str):

    df.sort_values(by=probability_col, ascending=False, inplace=True)

    subset = df[df[predicted_col] == True]

    rows = []
    for group in np.array_split(subset, 10):
        score = sklearn.metrics.accuracy_score(group[actual_col].tolist(),
                                                   group[predicted_col].tolist(),
                                                   normalize=False)

        rows.append({'NumCases': len(group), 'NumCorrectPredictions': score})

    lift = pd.DataFrame(rows)

    #Cumulative Gains Calculation
    lift['RunningCorrect'] = lift['NumCorrectPredictions'].cumsum()
    lift['PercentCorrect'] = lift.apply(
        lambda x: (100 / lift['NumCorrectPredictions'].sum()) * x['RunningCorrect'], axis=1)
    lift['CumulativeCorrectBestCase'] = lift['NumCases'].cumsum()
    lift['PercentCorrectBestCase'] = lift['CumulativeCorrectBestCase'].apply(
        lambda x: 100 if (100 / lift['NumCorrectPredictions'].sum()) * x > 100 else (100 / lift[
            'NumCorrectPredictions'].sum()) * x)
    lift['AvgCase'] = lift['NumCorrectPredictions'].sum() / len(lift)
    lift['CumulativeAvgCase'] = lift['AvgCase'].cumsum()
    lift['PercentAvgCase'] = lift['CumulativeAvgCase'].apply(
        lambda x: (100 / lift['NumCorrectPredictions'].sum()) * x)

    #Lift Chart
    lift['NormalisedPercentAvg'] = 1
    lift['NormalisedPercentWithModel'] = lift['PercentCorrect'] / lift['PercentAvgCase']

    return lift

要绘制累积收益图表,您可以使用下面的代码。

    import matplotlib.pyplot as plt
    def plot_cumulative_gains(lift: pd.DataFrame):
        fig, ax = plt.subplots()
        fig.canvas.draw()
    
        handles = []
        handles.append(ax.plot(lift['PercentCorrect'], 'r-', label='Percent Correct Predictions'))
        handles.append(ax.plot(lift['PercentCorrectBestCase'], 'g-', label='Best Case (for current model)'))
        handles.append(ax.plot(lift['PercentAvgCase'], 'b-', label='Average Case (for current model)'))
        ax.set_xlabel('Total Population (%)')
        ax.set_ylabel('Number of Respondents (%)')
    
        ax.set_xlim([0, 9])
        ax.set_ylim([10, 100])
    
        labels = [int((label+1)*10) for label in [float(item.get_text()) for item in ax.get_xticklabels()]]
    
        ax.set_xticklabels(labels)
    
        fig.legend(handles, labels=[h[0].get_label() for h in handles])
        fig.show()

可视化升力:

    def plot_lift_chart(lift: pd.DataFrame):
        plt.figure()
        plt.plot(lift['NormalisedPercentAvg'], 'r-', label='Normalised \'response rate\' with no model')
        plt.plot(lift['NormalisedPercentWithModel'], 'g-', label='Normalised \'response rate\' with using model')
        plt.legend()
        plt.show()

结果如下:

我发现这些网站可供参考:

编辑:

我发现 MS link 的描述有些误导,但 Paul Te Braak link 提供的信息非常丰富。回复评论;

对于上面的累积收益图表,所有计算均基于该特定模型的准确性。正如 Paul Te Braak link 指出的那样,我的模型的预测准确率如何才能达到 100%(图表中的红线)?最好的情况(绿线)是我们能以多快的速度达到红线在整个人口过程中达到的相同准确度(例如,我们的最佳累积收益情况)。蓝色是如果我们只是随机选择总体中每个样本的分类。因此,累积收益和提升图 纯粹是 用于了解该模型(并且仅该模型)如何在我不打算与整个人群互动的情况下给我更多的影响.

我使用累积收益图表的一个场景是针对欺诈案件,我想知道我们基本上可以忽略或优先考虑多少应用程序(因为我知道模型可以尽可能地预测它们)前 X%。在那种情况下,对于 'average model' 我改为从真实的无序数据集中选择分类(以显示现有应用程序的处理方式,以及如何使用模型 - 我们可以优先考虑应用程序类型)。

因此,要比较模型,请坚持使用 ROC/AUC,一旦您对所选模型感到满意,请使用累积增益/提升图查看它如何响应数据。

您可以使用 scikit-plot 包来完成繁重的工作。

skplt.metrics.plot_cumulative_gain(y_test, predicted_probas)

例子

# The usual train-test split mumbo-jumbo
from sklearn.datasets import load_breast_cancer
from sklearn.model_selection import train_test_split
from sklearn.naive_bayes import GaussianNB

X, y = load_breast_cancer(return_X_y=True)
X_train, X_test, y_train, y_test = train_test_split(X, y, 
test_size=0.33)
nb = GaussianNB()
nb.fit(X_train, y_train)
predicted_probas = nb.predict_proba(X_test)

# The magic happens here
import matplotlib.pyplot as plt
import scikitplot as skplt
skplt.metrics.plot_cumulative_gain(y_test, predicted_probas)
plt.show()

这应该会产生如下图:

您可以使用 kds 软件包。

对于累积收益图:

# pip install kds
import kds
kds.metrics.plot_cumulative_gain(y_test, y_prob)

例子

# REPRODUCABLE EXAMPLE
# Load Dataset and train-test split
from sklearn.datasets import load_iris
from sklearn.model_selection import train_test_split
from sklearn import tree

X, y = load_iris(return_X_y=True)
X_train, X_test, y_train, y_test = train_test_split(X, y, 
test_size=0.33,random_state=3)
clf = tree.DecisionTreeClassifier(max_depth=1,random_state=3)
clf = clf.fit(X_train, y_train)
y_prob = clf.predict_proba(X_test)


# CUMMULATIVE GAIN PLOT
import kds
kds.metrics.plot_cumulative_gain(y_test, y_prob[:,1])

向导曲线 将为模型提供最佳曲线。

免责声明:我是这个包的作者