-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathNode.m
340 lines (299 loc) · 12.6 KB
/
Node.m
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
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
classdef Node < handle
% The class for Node objects.
% This is an abstract class that can not be instantiated
properties
requiredHeight
% The height required for the node to be valid. Default: 1
lookupName = {}
% The name used to lookup the node type to determine the
% available types of nodes
childrenKeys = {}
% A cell of the name of the child or children the node has
returnType
% The expected return type of the node
template
% The template object for guiding the construction of the tree
tags = {}
% A cell of tags.
requiredTags = {}
% A cell of required tags.
excludedTags = {}
% A cell of tags that should be excluded.
end
methods
function node = Node(returnType, requiredHeight)
% The constructor of the Node object
% Arguments:
% returnType: The type it is expected to return
% requiredHeight: The height required for the node to
% be valid. Default value: 1
if nargin >= 1
node.returnType = returnType;
end
if nargin >= 2
node.requiredHeight = requiredHeight;
else
node.requiredHeight = 1;
end
end
function setTemplate(node, template)
if ~isa(template, 'Template')
error("Not a template object.");
end
node.template = template;
end
function initChildren(node, childrenKeys)
% Initialize the children dictionary (containers.Map) object
% with predefined keys.
% Arguments:
% childrenKeys: A cell of the name of the child or children
% the node has
node.childrenKeys = childrenKeys;
end
function appendLookupName(node, name)
node.lookupName{end+1} = name;
end
function lookupName = getLookupName(node)
lookupName = join(node.lookupName, '.');
lookupName = lookupName{1};
end
function grow(node, maxHeight)
% Grow the tree to the maximum suggested height.
if maxHeight < node.requiredHeight
error("Not enough height.");
end
n = length(node.childrenKeys);
% Find a node for each child
for i=1:n
childrenKey = node.childrenKeys{i};
while true
node.(childrenKey) = ...
node.template.getNode( ...
node.lookupName, childrenKey, maxHeight-1, true, ...
128, 0, node.tags);
if strcmp(childrenKey, 'rhs') && isa(node.(childrenKey), 'Value')
% If the variable does not wish to have a value at
% the right-hand side (valuesType == 0)
if node.lhs.valuesType == 0
continue;
end
end
break;
end
node.(childrenKey).addRequiredTags(node.requiredTags);
end
% Recursively grow the nodes
for i=1:n
childNode = node.(node.childrenKeys{i});
childNode.grow(maxHeight-1);
end
% Initialize the nodes
for i=1:n
if strcmp(node.childrenKeys{i}, 'rhs') && ...
isa(node.('rhs'), 'Value')
lhsNode = node.lhs;
rhsNode = node.rhs;
% Create the right type of Constant
if lhsNode.valuesType == 1
% Discrete variables
rhsNode = EnumeratedValue();
rhsNode.init( ...
lhsNode.fieldName, ...
lhsNode.allowedValues, ...
lhsNode.returnType);
elseif lhsNode.valuesType == 2
% Continuous variables
rhsNode = BoundedValue();
rhsNode.init( ...
lhsNode.fieldName, ...
lhsNode.lowerBound, ...
lhsNode.upperBound, ...
lhsNode.returnType);
else
error("Unknown valuesType.");
end
node.(node.childrenKeys{i}) = rhsNode;
else
childNode = node.(node.childrenKeys{i});
childNode.init();
end
end
end
function depth = getDepth(node)
n = length(node.childrenKeys);
if n == 0
depth = 1;
else
maxDepth = -1;
for i=1:n
childrenKey = node.childrenKeys{i};
depth = node.(childrenKey).getDepth();
if depth > maxDepth
maxDepth = depth;
end
end
depth = maxDepth + 1;
end
end
function nodes = getChildrenNodes(node)
n = length(node.childrenKeys);
nodes = cell(n, 1);
for i=1:n
nodes{i} = node.(node.childrenKeys{i});
end
end
function nodes = getNodesByDistance(node, distance)
if distance == 0
nodes = {node};
elseif distance == 1
nodes = node.getChildrenNodes();
else
nodes = {};
childrenNodes = node.getChildrenNodes();
n = length(node.childrenKeys);
for i=1:n
retNodes = childrenNodes{i}.getNodesByDistance(distance-1);
if ~isempty(retNodes)
nodes = cat(1, [nodes(:)', retNodes(:)']);
end
end
end
end
function success = crossover(nodeA, nodeB, ~)
success = false;
depthA = nodeA.getDepth();
depthB = nodeB.getDepth();
maxAttempts = depthA * depthB;
for attempt=1:maxAttempts
% Randomly select a node from tree A.
randDistanceA = randi([0, depthA-1]);
nodesA = nodeA.getNodesByDistance(randDistanceA);
selectedNodeIdxA = randi([1, length(nodesA)]);
selectedNodeA = nodesA{selectedNodeIdxA};
if isempty(selectedNodeA.childrenKeys)
% The node does not contain any child, move on to the
% next attempt.
continue;
end
numChildren = length(selectedNodeA.childrenKeys);
randomChildIdx = randi([1, numChildren]);
randomChildKey = selectedNodeA.childrenKeys{randomChildIdx};
% Randomly select a node from tree B.
randDistanceB = randi([0, depthB-1]);
nodesB = nodeB.getNodesByDistance(randDistanceB);
selectedNodeIdxB = randi([1, length(nodesB)]);
selectedNodeB = nodesB{selectedNodeIdxB};
% Check if selected node B is a valid node for becoming
% the child of A.
allowedNodes = nodeA.template.getNodes( ...
selectedNodeA.lookupName, randomChildKey, true, 128, 0);
numAllowedNodes = length(allowedNodes);
allowedHeight = depthA - randDistanceA - 1;
for i=1:numAllowedNodes
if strcmp(class(selectedNodeB), class(allowedNodes{i}))
% Check whether the height satisfies the
% requirements
if selectedNodeB.getDepth() > allowedHeight
% Break the loop without setting success to
% true.
break;
end
% Check if it is a valid crossover in terms of the
% requiredTags
if ~Template.checkTagsValidity( ...
selectedNodeA.tags, ...
selectedNodeB.requiredTags, ...
selectedNodeB.excludedTags)
break;
end
% Verified the node is valid
selectedNodeA.(randomChildKey) = ...
deepcopy(selectedNodeB);
success = true;
break;
end
end
if success
break;
end
end
end
function mutate(node, opt, maxHeight)
% Mutate the node. The default behaviors for nodes. Can be
% overridden if needed.
% The mutation is done by replacing regrowing the current
% node. The node itself (eg. the operator >=) will not be
% changed. Only the children is changed.
% Arguments:
% opt: A struct containing the following options
% - maxHeight: The maximum height allowed
% - mutationProb: The rate of mutation. For example, 0.02
% for 2%.
% - mutationStd: [Optional] A struct containing the
% mutation standard deviation of mutation for values
% or signals. For each entry, it should contain the
% name of the related environment variable and the
% standard deviation of mutation. For example,
% mutationStd.RSI = 1.
% maxHeight: The maximum height allowed. No need to specify.
% as the function will automatically populate the
% variable with values from the opt object
if nargin < 3
maxHeight = opt.maxHeight;
end
if rand() < opt.mutationProb
% The current node is chosen to mutate. Then try to grow
try
node.grow(opt.maxHeight-1);
catch
end
else
n = length(node.childrenKeys);
for i=1:n
childrenKey = node.childrenKeys{i};
childNode = node.(childrenKey);
childNode.mutate(opt, maxHeight-1);
end
end
end
function appendTags(node, tags)
% Append tags to the current node.
% Note:
% Only tags that are not present in the cell will be added.
% Arguments:
% tags: A cell of tags.
node.tags = ...
unique(cat(1, [node.tags(:)', tags(:)']));
end
function addRequiredTags(node, tags)
% Add required tags
% Note:
% Only tags that are not present in the cell will be added.
% Arguments:
% tags: A cell of tags that needs to be the prerequisite.
node.requiredTags = ...
unique(cat(1, [node.requiredTags(:)', tags(:)']));
end
function addExcludedTags(node, tags)
% Add excluded tags
% Note:
% Only tags that are not present in the cell will be added.
% Arguments:
% tags: A cell of tags that needs to be the excluded.
node.excludedTags = ...
unique(cat(1, [node.excludedTags(:)', tags(:)']));
end
end
methods(Abstract)
init(node)
% Initialize the node.
output = exec(node, env)
% Execute the current node.
summary(node, level)
% Generate the pseudocode of the current tree
% Arguments:
% level: The indentation level. Starts from 1. A 0 indicates
% there should not be a new line at the beginning of the
% current level.
end
end