Free cookie consent management tool by TermsFeed Policy Generator

source: branches/OKBJavaConnector/ECJClient/src/ec/gp/GPFunctionSet.java @ 10677

Last change on this file since 10677 was 6152, checked in by bfarka, 14 years ago

added ecj and custom statistics to communicate with the okb services #1441

File size: 12.5 KB
Line 
1/*
2  Copyright 2006 by Sean Luke
3  Licensed under the Academic Free License version 3.0
4  See the file "LICENSE" for more information
5*/
6
7
8package ec.gp;
9import java.io.*;
10import ec.*;
11import ec.util.*;
12import java.util.*;
13
14/*
15 * GPFunctionSet.java
16 *
17 * Created: Wed Oct 13 22:35:06 1999
18 * By: Sean Luke
19 */
20
21/**
22 * GPFunctionSet is a Clique which represents a set of GPNode prototypes
23 * forming a standard function set for forming certain trees in individuals.
24 * GPFunctionSets instances have unique names with which they're referenced by
25 * GPTreeConstraints objects indicating that they're used for certain trees.
26 * GPFunctionSets store their GPNode Prototypes in three hashtables,
27 * one for all nodes, one for nonterminals, and one for terminals.  Each
28 * hashed item is an array of GPNode objects,
29 * hashed by the return type of the GPNodes in the array.
30 *
31 * GPFunctionSets also contain prototypical GPNode nodes which they
32 * clone to form their arrays.
33
34 <p><b>Parameters</b><br>
35 <table>
36 <tr><td valign=top><i>base</i>.<tt>name</tt><br>
37 <font size=-1>String</font></td>
38 <td valign=top>(name of function set.  Must be different from other function set instances)</td></tr>
39
40 <tr><td valign=top><i>base</i>.<tt>size</tt><br>
41 <font size=-1>int &gt;= 1</font></td>
42 <td valign=top>(number of functions in the function set)</td></tr>
43
44 <tr><td valign=top><i>base</i>.<tt>func.</tt><i>n</i><br>
45 <font size=-1>classname, inherits and != ec.gp.GPNode</font></td>
46 <td valign=top>(class of function node <i>n</i> in the set)</td></tr>
47
48 </table>
49
50 <p><b>Parameter bases</b><br>
51 <table>
52 <tr><td valign=top><i>base</i>.<tt>func.</tt><i>n</i></td>
53 <td>function node <i>n</i></td></tr>
54 </table>
55 
56 *
57 * @author Sean Luke
58 * @version 1.0
59 */
60
61public class GPFunctionSet implements Clique
62    {
63    public final static String P_NAME = "name";
64    public final static String P_FUNC = "func";
65    public final static String P_SIZE = "size";
66
67    /** Name of the GPFunctionSet */
68    public String name;
69
70    /** The nodes that our GPTree can use: arrays of nodes hashed by type. */
71    public Hashtable nodes_h;
72    /** The nodes that our GPTree can use: nodes[type][thenodes]. */
73    public GPNode[][] nodes;
74    /** The nonterminals our GPTree can use: arrays of nonterminals hashed by type. */
75    public Hashtable nonterminals_h;
76    /** The nonterminals our GPTree can use: nonterminals[type][thenodes]. */
77    public GPNode[][] nonterminals;
78    /** The terminals our GPTree can use: arrays of terminals hashed by type. */
79    public Hashtable terminals_h;
80    /** The terminals our GPTree can use: terminals[type][thenodes]. */
81    public GPNode[][] terminals;
82
83
84    // some convenience methods which speed up various kinds
85    // of mutation operators
86
87    /** The nodes that our GPTree can use, hashed by name(). */
88    public Hashtable nodesByName;
89
90    /** Nodes == a given arity, that is: nodesByArity[type][arity][thenodes] */
91    public GPNode[][][]nodesByArity;
92
93    /** Nonterminals <= a given arity, that is: nonterminalsUnderArity[type][arity][thenodes] --
94        this will be O(n^2).  Obviously, the number of nonterminals at arity slot 0 is 0.*/
95    public GPNode[][][]nonterminalsUnderArity;
96
97    /** Nonterminals >= a given arity, that is: nonterminalsOverArity[type][arity][thenodes] --
98        this will be O(n^2).  Obviously, the number of nonterminals at arity slot 0 is all the
99        nonterminals of that type. */
100    public GPNode[][][]nonterminalsOverArity;
101   
102    /** Returns the name. */
103    public String toString() { return name; }
104
105
106    /** Sets up all the GPFunctionSet, loading them from the parameter
107        file.  This must be called before anything is called which refers
108        to a type by name. */
109
110    /** Sets up the arrays based on the hashtables */
111
112    public void postProcessFunctionSet()
113        {
114        nodes = new GPNode[nodes_h.size()][];
115        terminals = new GPNode[terminals_h.size()][];
116        nonterminals = new GPNode[nonterminals_h.size()][];
117
118        Enumeration e = nodes_h.keys();
119        while(e.hasMoreElements())
120            {
121            GPType gpt = (GPType)(e.nextElement());
122            GPNode[] gpfi = (GPNode[])(nodes_h.get(gpt));
123            nodes[gpt.type] = gpfi;
124            }
125        e = nonterminals_h.keys();
126        while(e.hasMoreElements())
127            {
128            GPType gpt = (GPType)(e.nextElement());
129            GPNode[] gpfi = (GPNode[])(nonterminals_h.get(gpt));
130            nonterminals[gpt.type] = gpfi;
131            }
132        e = terminals_h.keys();
133        while(e.hasMoreElements())
134            {
135            GPType gpt = (GPType)(e.nextElement());
136            GPNode[] gpfi = (GPNode[])(terminals_h.get(gpt));
137            terminals[gpt.type] = gpfi;
138            }
139
140        // set up arity-based arrays
141        // first, determine the maximum arity
142        int max_arity=0;
143        for(int x=0;x<nodes.length;x++)
144            for(int y=0;y<nodes[x].length;y++)
145                if (max_arity < nodes[x][y].children.length)
146                    max_arity = nodes[x][y].children.length;
147
148        // next set up the == array
149        nodesByArity = new GPNode[nodes.length][max_arity+1][];
150        for(int x=0;x<nodes.length;x++)
151            for(int a = 0; a <= max_arity; a++)
152                {
153                // how many nodes do we have?
154                int num_of_a = 0;
155                for(int y=0;y<nodes[x].length;y++)
156                    if (nodes[x][y].children.length == a) num_of_a++;
157                // allocate and fill
158                nodesByArity[x][a] = new GPNode[num_of_a];
159                int cur_a = 0;
160                for(int y=0;y<nodes[x].length;y++)
161                    if (nodes[x][y].children.length == a )
162                        nodesByArity[x][a][cur_a++] = nodes[x][y];
163                }
164
165        // now set up the <= nonterminals array
166        nonterminalsUnderArity = new GPNode[nonterminals.length][max_arity+1][];
167        for(int x=0;x<nonterminals.length;x++)
168            for (int a = 0;a <= max_arity; a++)
169                {
170                // how many nonterminals do we have?
171                int num_of_a = 0;
172                for(int y=0;y<nonterminals[x].length;y++)
173                    if (nonterminals[x][y].children.length <= a) num_of_a++;
174                // allocate and fill
175                nonterminalsUnderArity[x][a] = new GPNode[num_of_a];
176                int cur_a = 0;
177                for(int y=0;y<nonterminals[x].length;y++)
178                    if (nonterminals[x][y].children.length <= a )
179                        nonterminalsUnderArity[x][a][cur_a++] = nonterminals[x][y];
180                }
181
182
183
184        // now set up the >= nonterminals array
185        nonterminalsOverArity = new GPNode[nonterminals.length][max_arity+1][];
186        for(int x=0;x<nonterminals.length;x++)
187            for (int a = 0;a <= max_arity; a++)
188                {
189                // how many nonterminals do we have?
190                int num_of_a = 0;
191                for(int y=0;y<nonterminals[x].length;y++)
192                    if (nonterminals[x][y].children.length >= a) num_of_a++;
193                // allocate and fill
194                nonterminalsOverArity[x][a] = new GPNode[num_of_a];
195                int cur_a = 0;
196                for(int y=0;y<nonterminals[x].length;y++)
197                    if (nonterminals[x][y].children.length >= a )
198                        nonterminalsOverArity[x][a][cur_a++] = nonterminals[x][y];
199                }
200        }
201
202
203
204
205    /** Must be done <i>after</i> GPType and GPNodeConstraints have been set up */
206
207    public void setup(final EvolutionState state, final Parameter base)
208        {
209        // What's my name?
210        name = state.parameters.getString(base.push(P_NAME),null);
211        if (name==null)
212            state.output.fatal("No name was given for this function set.",
213                base.push(P_NAME));
214        // Register me
215        GPFunctionSet old_functionset = (GPFunctionSet)(((GPInitializer)state.initializer).functionSetRepository.put(name,this));
216        if (old_functionset != null)
217            state.output.fatal("The GPFunctionSet \"" + name + "\" has been defined multiple times.", base.push(P_NAME));
218
219        // How many functions do I have?
220        int numFuncs = state.parameters.getInt(base.push(P_SIZE),null,1);
221        if (numFuncs < 1)
222            state.output.error("The GPFunctionSet \"" + name + "\" has no functions.",
223                base.push(P_SIZE));
224       
225        nodesByName = new Hashtable();
226
227        Parameter p = base.push(P_FUNC);
228        Vector tmp = new Vector();
229        for(int x = 0; x < numFuncs; x++)
230            {
231            // load
232            Parameter pp = p.push(""+x);
233            GPNode gpfi = (GPNode)(state.parameters.getInstanceForParameter(
234                    pp, null, GPNode.class));
235            gpfi.setup(state,pp);
236
237            // add to my collection
238            tmp.addElement(gpfi);
239                       
240            // Load into the nodesByName hashtable
241            GPNode[] nodes = (GPNode[])(nodesByName.get(gpfi.name()));
242            if (nodes == null)
243                nodesByName.put(gpfi.name(), new GPNode[] { gpfi });
244            else
245                {
246                // O(n^2) but uncommon so what the heck.
247                GPNode[] nodes2 = new GPNode[nodes.length + 1];
248                System.arraycopy(nodes, 0, nodes2, 0, nodes.length);
249                nodes2[nodes2.length - 1] = gpfi;
250                nodesByName.put(gpfi.name(), nodes2);
251                }
252            }
253
254        // Make my hash tables
255        nodes_h = new Hashtable();
256        terminals_h = new Hashtable();
257        nonterminals_h = new Hashtable();
258
259        // Now set 'em up according to the types in GPType
260
261        Enumeration e = ((GPInitializer)state.initializer).typeRepository.elements();
262        GPInitializer initializer = ((GPInitializer)state.initializer);
263        while(e.hasMoreElements())
264            {
265            GPType typ = (GPType)(e.nextElement());
266           
267            // make vectors for the type.
268            Vector nodes_v = new Vector();
269            Vector terminals_v = new Vector();
270            Vector nonterminals_v = new Vector();
271
272            // add GPNodes as appropriate to each vector
273            Enumeration v = tmp.elements();
274            while (v.hasMoreElements())
275                {
276                GPNode i = (GPNode)(v.nextElement());
277                if (typ.compatibleWith(initializer,i.constraints(initializer).returntype))
278                    {
279                    nodes_v.addElement(i);
280                    if (i.children.length == 0)
281                        terminals_v.addElement(i);
282                    else nonterminals_v.addElement(i);
283                    }
284                }
285
286            // turn nodes_h' vectors into arrays
287            GPNode[] ii = new GPNode[nodes_v.size()];
288            nodes_v.copyInto(ii);
289            nodes_h.put(typ,ii);
290
291            // turn terminals_h' vectors into arrays
292            ii = new GPNode[terminals_v.size()];
293            terminals_v.copyInto(ii);
294            terminals_h.put(typ,ii);
295
296            // turn nonterminals_h' vectors into arrays
297            ii = new GPNode[nonterminals_v.size()];
298            nonterminals_v.copyInto(ii);
299            nonterminals_h.put(typ,ii);
300            }
301
302        // I don't check to see if the generation mechanism will be valid here
303        // -- I check that in GPTreeConstraints, where I can do the weaker check
304        // of going top-down through functions rather than making sure that every
305        // single function has a compatible argument function (an unneccessary check)
306
307        state.output.exitIfErrors();  // because I promised when I called n.setup(...)
308
309        // postprocess the function set
310        postProcessFunctionSet();
311        }
312
313
314    /** Returns the function set for a given name.
315        You must guarantee that after calling functionSetFor(...) one or
316        several times, you call state.output.exitIfErrors() once. */
317
318    public static GPFunctionSet functionSetFor(final String functionSetName,
319        final EvolutionState state)
320        {
321        GPFunctionSet set = (GPFunctionSet)(((GPInitializer)state.initializer).functionSetRepository.get(functionSetName));
322        if (set==null)
323            state.output.error("The GP function set \"" + functionSetName + "\" could not be found.");
324        return set;
325        }
326
327
328    private void writeObject(ObjectOutputStream out) throws IOException
329        {
330        // this wastes an hashtable pointer, but what the heck.
331
332        out.defaultWriteObject();
333        }
334
335    private void readObject(ObjectInputStream in) throws IOException, ClassNotFoundException
336        {
337        in.defaultReadObject();
338        }
339    }
Note: See TracBrowser for help on using the repository browser.