org.epochx.ge.model.groovy.EvenParity.java Source code

Java tutorial

Introduction

Here is the source code for org.epochx.ge.model.groovy.EvenParity.java

Source

/*
 * Copyright 2007-2010 Tom Castle & Lawrence Beadle
 * Licensed under GNU General Public License
 * 
 * This file is part of EpochX: genetic programming software for research
 * 
 * EpochX is free software: you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation, either version 3 of the License, or
 * (at your option) any later version.
 * 
 * EpochX is distributed in the hope that it will be useful,
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 * GNU General Public License for more details.
 * 
 * You should have received a copy of the GNU General Public License
 * along with EpochX. If not, see <http://www.gnu.org/licenses/>.
 * 
 * The latest version is available from: http:/www.epochx.org
 */
package org.epochx.ge.model.groovy;

import org.apache.commons.lang.ArrayUtils;
import org.epochx.ge.model.GEModel;
import org.epochx.ge.representation.GECandidateProgram;
import org.epochx.representation.CandidateProgram;
import org.epochx.tools.eval.*;
import org.epochx.tools.grammar.Grammar;
import org.epochx.tools.util.BoolUtils;

/**
 * Grammar model for the even parity problems using a Groovy grammar.
 * 
 * <h4>Even parity problem</h4>
 * 
 * Given n binary inputValues, a program that solves the even-n-parity problem
 * will return true in all circumstances where an even number of the inputValues
 * are true (or 1), and return false whenever there is an odd number of true
 * inputValues.
 */
public class EvenParity extends GEModel {

    // Incomplete grammar requiring correct number of terminals to be added.
    public static final String GRAMMAR_FRAGMENT = "<prog> ::= <expr>\n" + "<expr> ::= <expr> <op> <expr> "
            + "| ( <expr> <op> <expr> ) " + "| <var> " + "| <pre-op> ( <var> )\n" + "<pre-op> ::= !\n"
            + "<op> ::= \"|\" | & | ^ \n" + "<var> ::= ";

    // Groovy interpreter for performing evaluation.
    private final GroovyInterpreter interpreter;

    // The names of the inputValues used in the grammar.
    private final String[] argNames;

    // The boolean input sequences.
    private final boolean[][] inputValues;

    /**
     * Constructs an EvenParity model for the given number of inputs.
     * 
     * @param noInputBits the number of inputs the even parity problem should be
     *        for
     */
    public EvenParity(final int noInputBits) {
        interpreter = new GroovyInterpreter();

        // Generate the input sequences.
        inputValues = BoolUtils.generateBoolSequences(noInputBits);

        // Determine the input argument names.
        argNames = new String[noInputBits];
        for (int i = 0; i < noInputBits; i++) {
            argNames[i] = "d" + i;
        }

        // Complete the grammar string and construct grammar instance.
        setGrammar(new Grammar(getGrammarString()));
    }

    /**
     * Calculates the fitness score for the given program. The fitness of a
     * program for the even-parity problem is calculated by evaluating it
     * using each of the possible sets of input values. There are
     * <code>2^noInputBits</code> possible sets of inputs. The fitness of the
     * program is the quantity of those input sequences that the program
     * returned an incorrect response for. That is, a fitness value of
     * <code>0.0</code> indicates the program responded correctly for every
     * possible set of input values.
     * 
     * @param p {@inheritDoc}
     * @return the calculated fitness for the given program.
     */
    @Override
    public double getFitness(final CandidateProgram p) {
        final GECandidateProgram program = (GECandidateProgram) p;

        double score = 0;

        // Evaluate all possible inputValues.
        for (final boolean[] vars : inputValues) {
            // Convert to object array.
            final Boolean[] objVars = ArrayUtils.toObject(vars);

            Boolean result = null;
            try {
                result = (Boolean) interpreter.eval(program.getSourceCode(), argNames, objVars);
            } catch (final MalformedProgramException e) {
                // Assign worst possible fitness and stop evaluating.
                score = 0;
                break;
            }

            // Increment score for a correct response.
            if ((result != null) && (result == isEvenNoTrue(vars))) {
                score++;
            }
        }

        return inputValues.length - score;
    }

    /**
     * Constructs and returns the full grammar string for the even parity
     * problem with the correct number of input bits.
     * 
     * @return the grammar string for the even parity problem with the set
     *         number of input bits
     */
    public String getGrammarString() {
        final StringBuilder buffer = new StringBuilder(GRAMMAR_FRAGMENT);
        for (int i = 0; i < argNames.length; i++) {
            if (i > 0) {
                buffer.append(" | ");
            }
            buffer.append(argNames[i]);
        }
        buffer.append('\n');

        return buffer.toString();
    }

    /*
     * Calculate what the correct response should be for the given inputs.
     */
    private boolean isEvenNoTrue(final boolean[] input) {
        int noTrues = 0;

        for (final boolean b : input) {
            if (b) {
                noTrues++;
            }
        }

        return ((noTrues % 2) == 0);
    }
}