All Downloads are FREE. Search and download functionalities are using the official Maven repository.

weka.classifiers.bayes.net.search.local.LAGDHillClimber Maven / Gradle / Ivy

/*
 *   This program 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.
 *
 *   This program 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 this program.  If not, see .
 */

/*
 * LAGDHillClimber.java
 * Copyright (C) 2005-2012 Manuel Neubach
 * 
 */

package weka.classifiers.bayes.net.search.local;

import java.util.Collections;
import java.util.Enumeration;
import java.util.Vector;

import weka.classifiers.bayes.BayesNet;
import weka.core.Instances;
import weka.core.Option;
import weka.core.RevisionUtils;
import weka.core.Utils;

/**
 *  This Bayes Network learning algorithm uses a Look
 * Ahead Hill Climbing algorithm called LAGD Hill Climbing. Unlike Greedy Hill
 * Climbing it doesn't calculate a best greedy operation (adding, deleting or
 * reversing an arc) but a sequence of nrOfLookAheadSteps operations, which
 * leads to a network structure whose score is most likely higher in comparison
 * to the network obtained by performing a sequence of nrOfLookAheadSteps greedy
 * operations. The search is not restricted by an order on the variables (unlike
 * K2). The difference with B and B2 is that this hill climber also considers
 * arrows part of the naive Bayes structure for deletion.
 * 

* * * Valid options are: *

* *

 * -L <nr of look ahead steps>
 *  Look Ahead Depth
 * 
* *
 * -G <nr of good operations>
 *  Nr of Good Operations
 * 
* *
 * -P <nr of parents>
 *  Maximum number of parents
 * 
* *
 * -R
 *  Use arc reversal operation.
 *  (default false)
 * 
* *
 * -N
 *  Initial structure is empty (instead of Naive Bayes)
 * 
* *
 * -mbc
 *  Applies a Markov Blanket correction to the network structure, 
 *  after a network structure is learned. This ensures that all 
 *  nodes in the network are part of the Markov blanket of the 
 *  classifier node.
 * 
* *
 * -S [BAYES|MDL|ENTROPY|AIC|CROSS_CLASSIC|CROSS_BAYES]
 *  Score type (BAYES, BDeu, MDL, ENTROPY and AIC)
 * 
* * * * @author Manuel Neubach * @version $Revision: 10154 $ */ public class LAGDHillClimber extends HillClimber { /** for serialization */ static final long serialVersionUID = 7217437499439184344L; /** Number of Look Ahead Steps **/ int m_nNrOfLookAheadSteps = 2; /** Number of Good Operations per Step **/ int m_nNrOfGoodOperations = 5; /** * search determines the network structure/graph of the network * * @param bayesNet the network * @param instances the data to use * @throws Exception if something goes wrong */ @Override protected void search(BayesNet bayesNet, Instances instances) throws Exception { int k = m_nNrOfLookAheadSteps; // Number of Look Ahead Steps int l = m_nNrOfGoodOperations; // Number of Good Operations per step lookAheadInGoodDirectionsSearch(bayesNet, instances, k, l); } // search /** * lookAheadInGoodDirectionsSearch determines the network structure/graph of * the network with best score according to LAGD Hill Climbing * * @param bayesNet the network * @param instances the data to use * @param nrOfLookAheadSteps * @param nrOfGoodOperations * @throws Exception if something goes wrong */ protected void lookAheadInGoodDirectionsSearch(BayesNet bayesNet, Instances instances, int nrOfLookAheadSteps, int nrOfGoodOperations) throws Exception { System.out.println("Initializing Cache"); initCache(bayesNet, instances); while (nrOfLookAheadSteps > 1) { System.out.println("Look Ahead Depth: " + nrOfLookAheadSteps); boolean legalSequence = true; double sequenceDeltaScore = 0; Operation[] bestOperation = new Operation[nrOfLookAheadSteps]; bestOperation = getOptimalOperations(bayesNet, instances, nrOfLookAheadSteps, nrOfGoodOperations); for (int i = 0; i < nrOfLookAheadSteps; i++) { if (bestOperation[i] == null) { legalSequence = false; } else { sequenceDeltaScore += bestOperation[i].m_fDeltaScore; } } while (legalSequence && sequenceDeltaScore > 0) { System.out.println("Next Iteration.........................."); for (int i = 0; i < nrOfLookAheadSteps; i++) { performOperation(bayesNet, instances, bestOperation[i]); } bestOperation = getOptimalOperations(bayesNet, instances, nrOfLookAheadSteps, nrOfGoodOperations); sequenceDeltaScore = 0; for (int i = 0; i < nrOfLookAheadSteps; i++) { if (bestOperation[i] != null) { System.out.println(bestOperation[i].m_nOperation + " " + bestOperation[i].m_nHead + " " + bestOperation[i].m_nTail); sequenceDeltaScore += bestOperation[i].m_fDeltaScore; } else { legalSequence = false; } System.out.println("DeltaScore: " + sequenceDeltaScore); } } --nrOfLookAheadSteps; } /** last steps with greedy HC **/ Operation oOperation = getOptimalOperation(bayesNet, instances); while ((oOperation != null) && (oOperation.m_fDeltaScore > 0)) { performOperation(bayesNet, instances, oOperation); System.out.println("Performing last greedy steps"); oOperation = getOptimalOperation(bayesNet, instances); } // free up memory m_Cache = null; } // lookAheadInGoodDirectionsSearch /** * getAntiOperation determines the Operation, which is needed to cancel * oOperation * * @param oOperation Operation to cancel * @return antiOperation to oOperation * @throws Exception if something goes wrong */ protected Operation getAntiOperation(Operation oOperation) throws Exception { if (oOperation.m_nOperation == Operation.OPERATION_ADD) { return (new Operation(oOperation.m_nTail, oOperation.m_nHead, Operation.OPERATION_DEL)); } else { if (oOperation.m_nOperation == Operation.OPERATION_DEL) { return (new Operation(oOperation.m_nTail, oOperation.m_nHead, Operation.OPERATION_ADD)); } else { return (new Operation(oOperation.m_nHead, oOperation.m_nTail, Operation.OPERATION_REVERSE)); } } } // getAntiOperation /** * getGoodOperations determines the nrOfGoodOperations best Operations, which * are considered for the calculation of an optimal operationsequence * * @param bayesNet Bayes network to apply operation on * @param instances data set to learn from * @param nrOfGoodOperations number of good operations to consider * @return good operations to consider * @throws Exception if something goes wrong **/ protected Operation[] getGoodOperations(BayesNet bayesNet, Instances instances, int nrOfGoodOperations) throws Exception { Operation[] goodOperations = new Operation[nrOfGoodOperations]; for (int i = 0; i < nrOfGoodOperations; i++) { goodOperations[i] = getOptimalOperation(bayesNet, instances); if (goodOperations[i] != null) { m_Cache.put(goodOperations[i], -1E100); } else { i = nrOfGoodOperations; } } for (int i = 0; i < nrOfGoodOperations; i++) { if (goodOperations[i] != null) { if (goodOperations[i].m_nOperation != Operation.OPERATION_REVERSE) { m_Cache.put(goodOperations[i], goodOperations[i].m_fDeltaScore); } else { m_Cache .put( goodOperations[i], goodOperations[i].m_fDeltaScore - m_Cache.m_fDeltaScoreAdd[goodOperations[i].m_nHead][goodOperations[i].m_nTail]); } } else { i = nrOfGoodOperations; } } return goodOperations; } // getGoodOperations /** * getOptimalOperations determines an optimal operationsequence in respect of * the parameters nrOfLookAheadSteps and nrOfGoodOperations * * @param bayesNet Bayes network to apply operation on * @param instances data set to learn from * @param nrOfLookAheadSteps number of lood ahead steps to use * @param nrOfGoodOperations number of good operations to consider * @return optimal sequence of operations in respect to nrOfLookAheadSteps and * nrOfGoodOperations * @throws Exception if something goes wrong **/ protected Operation[] getOptimalOperations(BayesNet bayesNet, Instances instances, int nrOfLookAheadSteps, int nrOfGoodOperations) throws Exception { if (nrOfLookAheadSteps == 1) { // Abbruch der Rekursion Operation[] bestOperation = new Operation[1]; bestOperation[0] = getOptimalOperation(bayesNet, instances); return (bestOperation); // Abbruch der Rekursion } else { double bestDeltaScore = 0; double currentDeltaScore = 0; Operation[] bestOperation = new Operation[nrOfLookAheadSteps]; Operation[] goodOperations = new Operation[nrOfGoodOperations]; Operation[] tempOperation = new Operation[nrOfLookAheadSteps - 1]; goodOperations = getGoodOperations(bayesNet, instances, nrOfGoodOperations); for (int i = 0; i < nrOfGoodOperations; i++) { if (goodOperations[i] != null) { performOperation(bayesNet, instances, goodOperations[i]); tempOperation = getOptimalOperations(bayesNet, instances, nrOfLookAheadSteps - 1, nrOfGoodOperations); // rekursiver Abstieg currentDeltaScore = goodOperations[i].m_fDeltaScore; for (int j = 0; j < nrOfLookAheadSteps - 1; j++) { if (tempOperation[j] != null) { currentDeltaScore += tempOperation[j].m_fDeltaScore; } } performOperation(bayesNet, instances, getAntiOperation(goodOperations[i])); if (currentDeltaScore > bestDeltaScore) { bestDeltaScore = currentDeltaScore; bestOperation[0] = goodOperations[i]; for (int j = 1; j < nrOfLookAheadSteps; j++) { bestOperation[j] = tempOperation[j - 1]; } } } else { i = nrOfGoodOperations; } } return (bestOperation); } } // getOptimalOperations /** * Sets the max number of parents * * @param nMaxNrOfParents the max number of parents */ @Override public void setMaxNrOfParents(int nMaxNrOfParents) { m_nMaxNrOfParents = nMaxNrOfParents; } /** * Gets the max number of parents. * * @return the max number of parents */ @Override public int getMaxNrOfParents() { return m_nMaxNrOfParents; } /** * Sets the number of look-ahead steps * * @param nNrOfLookAheadSteps the number of look-ahead steps */ public void setNrOfLookAheadSteps(int nNrOfLookAheadSteps) { m_nNrOfLookAheadSteps = nNrOfLookAheadSteps; } /** * Gets the number of look-ahead steps * * @return the number of look-ahead step */ public int getNrOfLookAheadSteps() { return m_nNrOfLookAheadSteps; } /** * Sets the number of "good operations" * * @param nNrOfGoodOperations the number of "good operations" */ public void setNrOfGoodOperations(int nNrOfGoodOperations) { m_nNrOfGoodOperations = nNrOfGoodOperations; } /** * Gets the number of "good operations" * * @return the number of "good operations" */ public int getNrOfGoodOperations() { return m_nNrOfGoodOperations; } /** * Returns an enumeration describing the available options. * * @return an enumeration of all the available options. */ @Override public Enumeration




© 2015 - 2025 Weber Informatics LLC | Privacy Policy