From 5627040e601651e906132e6d0c5bb9d00664ebfc Mon Sep 17 00:00:00 2001 From: Mihail Guschin Date: Tue, 15 Mar 2016 16:37:58 +0300 Subject: [PATCH] 4 classes. ATTENTION! 4 classes. --- .gitignore | 4 + python/estimators.py | 84 ++++++++++++ python/utils.py | 308 +++++++++++++++++++++++++++++++++++++++++++ 3 files changed, 396 insertions(+) create mode 100644 .gitignore create mode 100644 python/estimators.py create mode 100644 python/utils.py diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..7d1b4eb --- /dev/null +++ b/.gitignore @@ -0,0 +1,4 @@ +*.idea +*.pyc +*.DS_Store +*.ipynb_checkpoints diff --git a/python/estimators.py b/python/estimators.py new file mode 100644 index 0000000..14c7a7c --- /dev/null +++ b/python/estimators.py @@ -0,0 +1,84 @@ +__author__ = 'mikhail91' + +import copy +import numpy + +class OneVsRestClassifier(object): + + def __init__(self, estimator): + """ + One vs Rest classifier based on an estimator. + :param estimator: estimator with fit, predict, predict_proba mathods. + :return: + """ + + self.estimator = estimator + self.estimators_ = {} + self.classes = [] + + def fit(self, X, y, sample_weight=None): + """ + Fit the classifier. + :param X: the estimator's format, data + :param y: array, shape = [n_samples], labels of classes [0, 1, 2, ..., n_classes - 1] + :param sample_weight: None, 'balanced' or array, shape = [n_samples], sample weights. + If 'balanced' sum of weights of positive and negative classes will be equal. + :return: + """ + + self.classes = numpy.unique(y) + + for one_class in self.classes: + + y_class = (y == one_class) * 1. + + estimator = copy.copy(self.estimator) + + if sample_weight is None: + + estimator.fit(X, y_class) + + elif sample_weight == 'balanced': + + weights = (y == one_class) * len(y) / ((y == one_class).sum()) + \ + (y != one_class) * len(y) / ((y != one_class).sum()) + + estimator.fit(X, y_class, sample_weight = weights) + + else: + + estimator.fit(X, y_class, sample_weight) + + self.estimators_[one_class] = estimator + + def predict_proba(self, X): + """ + Predict probabilities to belong to a class for the each class. + :param X: the estimator's format, data + :return: ndarray, shape = [n_samples, n_classes], probabiities. + """ + + probas = numpy.zeros((len(X), len(self.classes))) + + for num, one_class in enumerate(self.classes): + + one_proba = self.estimators_[one_class].predict_proba(X)[:, 1] + + probas[:, num] = one_proba + + return probas + + def predict(self, X): + """ + Predict classes. + :param X: the estimator's format, data + :return: array, shape = [n_samples], class labels [0, 1, 2, ..., n_classes - 1] + """ + + probas = self.predict_proba(X) + + predictions = probas.argmax(axis=1) + + return predictions + + diff --git a/python/utils.py b/python/utils.py new file mode 100644 index 0000000..93c7c1a --- /dev/null +++ b/python/utils.py @@ -0,0 +1,308 @@ +import numpy +import pandas +from sklearn.metrics import roc_curve, auc, roc_auc_score +import matplotlib.pyplot as plt +import matplotlib as mpl +from matplotlib import cm +from rep.utils import get_efficiencies +from rep.plotting import ErrorPlot + + +def __rolling_window(data, window_size): + """ + Rolling window: take window with definite size through the array + + :param data: array-like + :param window_size: size + :return: the sequence of windows + + Example: data = array(1, 2, 3, 4, 5, 6), window_size = 4 + Then this function return array(array(1, 2, 3, 4), array(2, 3, 4, 5), array(3, 4, 5, 6)) + """ + shape = data.shape[:-1] + (data.shape[-1] - window_size + 1, window_size) + strides = data.strides + (data.strides[-1],) + return numpy.lib.stride_tricks.as_strided(data, shape=shape, strides=strides) + + +def __cvm(subindices, total_events): + """ + Compute Cramer-von Mises metric. + Compared two distributions, where first is subset of second one. + Assuming that second is ordered by ascending + + :param subindices: indices of events which will be associated with the first distribution + :param total_events: count of events in the second distribution + :return: cvm metric + """ + # here we compute the same expression (using algebraic expressions for them). + n_subindices = float(len(subindices)) + subindices = numpy.array([0] + sorted(subindices) + [total_events], dtype='int') + # via sum of the first squares + summand1 = total_events * (total_events + 1) * (total_events + 0.5) / 3. / (total_events ** 3) + left_positions = subindices[:-1] + right_positions = subindices[1:] + + values = numpy.arange(len(subindices) - 1) + + summand2 = values * (right_positions * (right_positions + 1) - left_positions * (left_positions + 1)) / 2 + summand2 = summand2.sum() * 1. / (n_subindices * total_events * total_events) + + summand3 = (right_positions - left_positions) * values ** 2 + summand3 = summand3.sum() * 1. / (n_subindices * n_subindices * total_events) + + return summand1 + summand3 - 2 * summand2 + + +def compute_cvm(predictions, masses, n_neighbours=200, step=50): + """ + Computing Cramer-von Mises (cvm) metric on background events: take average of cvms calculated for each mass bin. + In each mass bin global prediction's cdf is compared to prediction's cdf in mass bin. + + :param predictions: array-like, predictions + :param masses: array-like, in case of Kaggle tau23mu this is reconstructed mass + :param n_neighbours: count of neighbours for event to define mass bin + :param step: step through sorted mass-array to define next center of bin + :return: average cvm value + """ + predictions = numpy.array(predictions) + masses = numpy.array(masses) + assert len(predictions) == len(masses) + + # First, reorder by masses + predictions = predictions[numpy.argsort(masses)] + + # Second, replace probabilities with order of probability among other events + predictions = numpy.argsort(numpy.argsort(predictions)) + + # Now, each window forms a group, and we can compute contribution of each group to CvM + cvms = [] + for window in __rolling_window(predictions, window_size=n_neighbours)[::step]: + cvms.append(__cvm(subindices=window, total_events=len(predictions))) + return numpy.mean(cvms) + + + +def labels_transform(labels): + + """ + Transform labels from shape = [n_samples] to shape = [n_samples, n_classes] + :param labels: array + :return: ndarray, transformed labels + """ + + classes = numpy.unique(labels) + + new_labels = numpy.zeros((len(labels), len(classes))) + for cl in classes: + new_labels[:, cl] = (labels == cl) * 1. + + return new_labels + + +def get_roc_curves(labels, probas, curve_labels, save_path=None, show=True): + """ + Creates roc curve for each class vs rest. + :param labels: array, shape = [n_samples], labels for the each class 0, 1, ..., n_classes - 1. + :param probas: ndarray, shape = [n_samples, n_classes], predicted probabilities. + :param curve_labels: array of strings , shape = [n_classes], labels of the curves. + :param save_path: string, path to a directory where the figure will saved. If None the figure will not be saved. + :param show: boolean, if true the figure will be displayed. + """ + + labels = labels_transform(labels) + + plt.figure(figsize=(10,7)) + + for num in range(probas.shape[1]): + + roc_auc = roc_auc_score(labels[:, num], probas[:, num]) + fpr, tpr, _ = roc_curve(labels[:, num], probas[:, num]) + + plt.plot(tpr, 1.-fpr, label=curve_labels[num] + ', %.4f' % roc_auc, linewidth=2) + + plt.title("ROC Curves", size=15) + plt.xlabel("Signal efficiency", size=15) + plt.ylabel("Background rejection", size=15) + plt.legend(loc='best',prop={'size':15}) + plt.xticks(numpy.arange(0, 1.01, 0.1), size=15) + plt.yticks(numpy.arange(0, 1.01, 0.1), size=15) + + + if save_path != None: + plt.savefig(save_path + "/overall_roc_auc.png") + + if show == True: + plt.show() + + plt.clf() + plt.close() + +def get_roc_auc_matrix(labels, probas, axis_labels, save_path=None, show=True): + + """ + Calculate class vs class roc aucs matrix. + :param labels: array, shape = [n_samples], labels for the each class 0, 1, ..., n_classes - 1. + :param probas: ndarray, shape = [n_samples, n_classes], predicted probabilities. + :param axis_labels: array of strings , shape = [n_classes], labels of the curves. + :param save_path: string, path to a directory where the figure will saved. If None the figure will not be saved. + :param show: boolean, if true the figure will be displayed. + :return: pandas.DataFrame roc_auc_matrix + """ + + labels = labels_transform(labels) + + # Calculate roc_auc_matrices + roc_auc_matrices = numpy.ones((probas.shape[1],probas.shape[1])) + + for first in range(probas.shape[1]): + for second in range(probas.shape[1]): + + if first == second: + continue + + weights = ((labels[:, first] != 0) + (labels[:, second] != 0)) * 1. + + roc_auc = roc_auc_score(labels[:, first], probas[:, first]/probas[:, second], sample_weight=weights) + + roc_auc_matrices[first, second] = roc_auc + + + # Save roc_auc_matrices + matrix = pandas.DataFrame(columns=axis_labels, index=axis_labels) + + for num in range(len(axis_labels)): + + matrix[axis_labels[num]] = roc_auc_matrices[num, :] + + if save_path != None: + matrix.to_csv(save_path + "/class_vs_class_roc_auc_matrix.csv") + + + # Plot roc_auc_matrices + inline_rc = dict(mpl.rcParams) + import seaborn as sns + plt.figure(figsize=(10,7)) + sns.set() + ax = plt.axes() + sns.heatmap(matrix, vmin=0.8, vmax=1., annot=True, fmt='.4f', ax=ax, cmap=cm.coolwarm) + plt.title('Particle vs particle roc aucs', size=15) + plt.xticks(size=15) + plt.yticks(size=15) + + if save_path != None: + plt.savefig(save_path + "/class_vs_class_roc_auc_matrix.png") + + if show == True: + plt.show() + + plt.clf() + plt.close() + + mpl.rcParams.update(mpl.rcParamsDefault) + mpl.rcParams.update(inline_rc) + + return matrix + +def get_roc_auc_ratio_matrix(matrix_one, matrix_two, save_path=None, show=True): + + """ + Divide matrix_one to matrix_two. + :param matrix_one: pandas.DataFrame with column 'Class' which contain class names. + :param matrix_two: pandas.DataFrame with column 'Class' which contain class names. + :param save_path: string, path to a directory where the figure will saved. If None the figure will not be saved. + :param show: boolean, if true the figure will be displayed. + :return: pandas.DataFrame roc_auc_ratio_matrix + """ + + # Calculate roc_auc_matrices + classes = list(matrix_one.index) + roc_auc_matrices = numpy.ones((len(classes), len(classes))) + + for first in range(len(classes)): + for second in range(len(classes)): + + roc_auc_one = matrix_one.loc[classes[first], classes[second]] + roc_auc_two = matrix_two.loc[classes[first], classes[second]] + roc_auc_matrices[first, second] = roc_auc_one / roc_auc_two + + # Save roc_auc_matrices + matrix = pandas.DataFrame(columns=classes, index=classes) + + for num in range(len(classes)): + + matrix[classes[num]] = roc_auc_matrices[num, :] + + if save_path != None: + matrix.to_csv(save_path + "/class_vs_class_roc_auc_rel_matrix.csv") + + # Plot roc_auc_matrices + from matplotlib import cm + inline_rc = dict(mpl.rcParams) + import seaborn as sns + plt.figure(figsize=(10,7)) + sns.set() + ax = plt.axes() + sns.heatmap(matrix, vmin=0.9, vmax=1.1, annot=True, fmt='.4f', ax=ax, cmap=cm.seismic) + plt.xticks(size=15) + plt.yticks(size=15) + plt.title('Particle vs particle roc aucs ratio', size=15) + + if save_path != None: + plt.savefig(save_path + "/class_vs_class_roc_auc_rel_matrix.png") + + if show == True: + plt.show() + + plt.clf() + plt.close() + + mpl.rcParams.update(mpl.rcParamsDefault) + mpl.rcParams.update(inline_rc) + + return matrix + + +def get_one_vs_one_roc_curves(labels, probas, curve_labels, save_path=None, show=True): + """ + Creates one vs one roc curves. + :param labels: array, shape = [n_samples], labels for the each class 0, 1, ..., n_classes - 1. + :param probas: ndarray, shape = [n_samples, n_classes], predicted probabilities. + :param curve_labels: array of strings , shape = [n_classes], labels of the curves. + :param save_path: string, path to a directory where the figure will saved. If None the figure will not be saved. + :param show: boolean, if true the figure will be displayed. + """ + + classes = numpy.unique(labels) + + for one_class, one_name in zip(classes, curve_labels): + + plt.figure(figsize=(10,7)) + + for two_class, two_name in zip(classes, curve_labels): + + if one_class == two_class: + continue + + weights = (labels == one_class) * 1. + (labels == two_class) * 1. + one_labels = (labels == one_class) * 1. + roc_auc = roc_auc_score(one_labels, probas[:, one_class] / probas[:, two_class], sample_weight=weights) + fpr, tpr, _ = roc_curve(one_labels, probas[:, one_class] / probas[:, two_class], sample_weight=weights) + + plt.plot(tpr, 1.-fpr, label=one_name + ' vs ' + two_name + ', %.4f' % roc_auc, linewidth=2) + + plt.title("ROC Curves", size=15) + plt.xlabel("Signal efficiency", size=15) + plt.ylabel("Background rejection", size=15) + plt.legend(loc='best',prop={'size':15}) + plt.xticks(numpy.arange(0, 1.01, 0.1), size=15) + plt.yticks(numpy.arange(0, 1.01, 0.1), size=15) + + + if save_path != None: + plt.savefig(save_path + "/" + one_name + "_vs_one_roc_auc.png") + + if show == True: + plt.show() + + plt.clf() + plt.close() \ No newline at end of file