-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrdgv3.cpp
92 lines (83 loc) · 2.87 KB
/
rdgv3.cpp
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
83
84
85
86
87
88
89
90
91
92
#include "rdgv3.hpp"
#include "common_funcs.hpp"
using std::abs;
using std::pow;
using std::make_tuple;
RDGv3::RDGv3() {}
RDGv3::RDGv3(const RDGv3& o) : RDGv2(o) {}
RDGv3::~RDGv3() {}
string RDGv3::info() {
return "Recursive Differential Grouping version 3 (" + sinfo() + ")";
}
string RDGv3::sinfo() {
return "RDG3";
}
tuple<vector<size_t>, vector<vector<size_t>>> RDGv3::run(BoundedObjectiveFunction<double>& fitf) {
auto x = new double[fitf.dim()];
for (int i = 0; i < fitf.dim(); i++) x[i] = fitf.x_bound_min(i);
auto xf = fitf(x);
auto sub1 = vector<size_t>(1, 0), sub2 = vector<size_t>(fitf.dim() - 1);
for (int i = 0; i < sub2.size(); i++) sub2[i] = i + 1;
auto xremain = vector<size_t>(1, 0);
auto seps = vector<size_t>();
auto nongroups = vector<vector<size_t>>();
unsigned int tn = fitf.dim() * tnf;
while (xremain.size() > 0) {
xremain = vector<size_t>();
auto sub1_a = interact(x, xf, sub1, sub2, xremain, fitf);
if (sub1_a.size() != sub1.size() && sub1_a.size() < tn) {
sub1 = sub1_a;
sub2 = xremain;
if (xremain.size() == 0) {
nongroups.push_back(sub1);
break;
}
} else {
if (sub1_a.size() == 1) {
seps.insert(seps.end(), sub1_a.begin(), sub1_a.end());
} else {
nongroups.push_back(sub1_a);
}
if (xremain.size() > 1) {
sub1 = vector<size_t>(1, xremain[0]);
xremain.erase(xremain.begin());
sub2 = xremain;
} else if (xremain.size() == 1) {
seps.push_back(xremain[0]);
break;
}
}
}
delete[] x;
return make_tuple(seps, nongroups);
}
vector<size_t> RDGv3::interact(double *a, double af, vector<size_t> sub1, vector<size_t> sub2, vector<size_t> &xremain, BoundedObjectiveFunction<double>& fitf) {
auto b = new double[fitf.dim()], c = new double[fitf.dim()], d = new double[fitf.dim()];
for (int i = 0; i < fitf.dim(); i++) b[i] = c[i] = a[i];
for (int i = 0; i < sub1.size(); i++) b[sub1[i]] = fitf.x_bound_max(sub1[i]);
auto bf = fitf(b);
auto delta1 = af - bf;
for (int i = 0; i < fitf.dim(); i++) d[i] = b[i];
for (int i = 0; i < sub2.size(); i++) c[sub2[i]] = d[sub2[i]] = (fitf.x_bound_max(sub2[i]) - fitf.x_bound_min(sub2[i])) / 2;
auto cf = fitf(c), df = fitf(d);
auto delta2 = cf - df;
delete[] b, delete[] c, delete[] d;
double epsilon = gamma(pow(fitf.dim(), 0.5) + 2) * (abs(af) + abs(bf) + abs(cf) + abs(df));
if (abs(delta1 - delta2) > epsilon) {
if (sub2.size() == 1) {
sub1 = vunion(sub1, sub2);
} else {
size_t k = floor(sub2.size() / 2);
auto sub2_1 = vector<size_t>();
for (int i = 0; i < k; i++) sub2_1.push_back(sub2[i]);
auto sub2_2 = vector<size_t>();
for (int i = 0; i < sub2.size() - k; i++) sub2_2.push_back(sub2[k + i]);
auto sub1_1 = interact(a, af, sub1, sub2_1, xremain, fitf);
auto sub1_2 = interact(a, af, sub1, sub2_2, xremain, fitf);
sub1 = vunion(sub1_1, sub1_2);
}
} else {
xremain.insert(xremain.end(), sub2.begin(), sub2.end());
}
return sub1;
}