-
Notifications
You must be signed in to change notification settings - Fork 39
/
Copy pathAdaBoostClassifier.py
83 lines (72 loc) · 3.25 KB
/
AdaBoostClassifier.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
"""
只针对2分类
自己实现的DecisionTreeClassifier没有实现 sample_weight参数
重点在AdaBoost, 使用sklearn的DecisionTreeClassifier作为基学习器
"""
import numpy as np
import copy
from sklearn.tree import DecisionTreeClassifier
from sklearn.ensemble import AdaBoostClassifier as sklearnAdaBoostClassifier
import sklearn.datasets as datasets
class AdaBoostClassifier:
def __init__(self, base_estimator=None, n_estimators=300,method='re-weighting'):
self.base_estimator = base_estimator
self.n_estimators = n_estimators
self.method=method
self.hs_ = []
self.epsilons_ = []
self.alphas_ = []
self.Ds_ = []
def fit(self, X, y):
m = X.shape[0]
self.Ds_.append(np.ones((m,)) / m)
for t in range(self.n_estimators):
ht = self.base_estimator
if self.method=='re-weighting':
ht.fit(X, y, self.Ds_[t])
elif self.method=='re-sampling':
sample_indices=np.random.choice(range(m),size=m,p=self.Ds_[t])
ht.fit(X[sample_indices],y[sample_indices])
y_pred = ht.predict(X).astype(np.int32)
valid_indices = (y != y_pred)
mask = np.ones((len(y),))
mask[valid_indices] = 0
epsilon_t = 1 - np.sum(self.Ds_[t] * mask)
if epsilon_t > 0.5:
break
self.hs_.append(copy.copy(ht))
self.epsilons_.append(epsilon_t)
alpha_t = 0.5 * np.log((1 - epsilon_t) / epsilon_t)
self.alphas_.append(alpha_t)
self.Ds_.append(self.Ds_[t] * np.exp(-alpha_t * y * y_pred))
self.Ds_[t + 1] = self.Ds_[t + 1] / np.sum(self.Ds_[t + 1])
@classmethod
def calc_epsilon(clf, D, y_target, y_pred):
return 1 - np.sum(D[y_target == y_pred])
def predict(self, X):
H=np.zeros((X.shape[0],))
for t in range(len(self.alphas_)):
H+=(self.alphas_[t]*self.hs_[t].predict(X))
return np.sign(H)
if __name__ == '__main__':
breast_data = datasets.load_breast_cancer()
X, y = breast_data.data, breast_data.target
y = 2 * y - 1
X_train, y_train = X[:200], y[:200]
X_test, y_test = X[200:], y[200:]
base_estimator=DecisionTreeClassifier(max_depth=1,random_state=False)
sklearn_decision_tree = DecisionTreeClassifier(max_depth=1)
sklearn_decision_tree.fit(X_train, y_train)
y_pred_decison_tree = sklearn_decision_tree.predict(X_test)
print('single decision tree:', len(y_test[y_pred_decison_tree == y_test]) * 1.0 / len(y_test))
print('tinyml:')
adaboost_clf = AdaBoostClassifier(n_estimators=100,base_estimator=base_estimator,method='re-weighting')
adaboost_clf.fit(X_train, y_train)
y_pred = adaboost_clf.predict(X_test)
print('adaboost y_pred:', len(y_test[y_pred == y_test]) * 1. / len(y_test))
print('sklearn:')
sklearn_adboost_clf = sklearnAdaBoostClassifier(n_estimators=100, random_state=False, algorithm='SAMME',
base_estimator=base_estimator)
sklearn_adboost_clf.fit(X_train, y_train)
sklearn_y_pred = sklearn_adboost_clf.predict(X_test)
print('sklearn adaboost y_pred:', len(y_test[y_test == sklearn_y_pred]) * 1. / len(y_test))