#region License Information /* HeuristicLab * Copyright (C) 2002-2008 Heuristic and Evolutionary Algorithms Laboratory (HEAL) * * This file is part of HeuristicLab. * * HeuristicLab 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. * * HeuristicLab 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 HeuristicLab. If not, see . */ #endregion using System; using System.Collections.Generic; using System.Text; using System.Xml; using System.Threading; namespace HeuristicLab.Core { /// /// Base class to represent an engine, which is an interpreter, holding the code, the data and /// the actual state, which is the runtime stack and a pointer onto the next operation. It represents /// one execution and can handle parallel executions. /// public abstract class EngineBase : ItemBase, IEngine { /// /// Field of the current instance that represent the operator graph. /// protected IOperatorGraph myOperatorGraph; /// /// Gets the current operator graph. /// public IOperatorGraph OperatorGraph { get { return myOperatorGraph; } } /// /// Field of the current instance that represent the global scope. /// protected IScope myGlobalScope; /// /// Gets the current global scope. /// public IScope GlobalScope { get { return myGlobalScope; } } private TimeSpan myExecutionTime; /// /// Gets or sets the execution time. /// /// Calls in the setter. public TimeSpan ExecutionTime { get { return myExecutionTime; } protected set { myExecutionTime = value; OnExecutionTimeChanged(); } } /// /// Field of the current instance that represent the execution stack. /// protected Stack myExecutionStack; /// /// Gets the current execution stack. /// public Stack ExecutionStack { get { return myExecutionStack; } } /// /// Flag of the current instance whether it is currently running. /// protected bool myRunning; /// /// Gets information whether the instance is currently running. /// public bool Running { get { return myRunning; } } /// /// Flag of the current instance whether it is canceled. /// protected bool myCanceled; /// /// Gets information whether the instance is currently canceled. /// public bool Canceled { get { return myCanceled; } } /// /// Gets information whether the instance has already terminated. /// public virtual bool Terminated { get { return ExecutionStack.Count == 0; } } /// /// Initializes a new instance of with a new global scope. /// /// Calls . protected EngineBase() { myOperatorGraph = new OperatorGraph(); myGlobalScope = new Scope("Global"); myExecutionStack = new Stack(); Reset(); } /// /// Copy constructor. /// /// /// protected EngineBase(EngineBase original, IDictionary clonedObjects) : base(original, clonedObjects) { this.myOperatorGraph = (IOperatorGraph)Auxiliary.Clone(original.OperatorGraph, clonedObjects); this.myGlobalScope = (IScope)Auxiliary.Clone(original.GlobalScope, clonedObjects); this.myExecutionTime = original.ExecutionTime; IOperation[] operations = new IOperation[original.ExecutionStack.Count]; original.ExecutionStack.CopyTo(operations, 0); for (int i = operations.Length - 1; i >= 0; i--) this.myExecutionStack.Push((IOperation)Auxiliary.Clone(operations[i], clonedObjects)); this.myRunning = original.Running; this.myCanceled = original.Canceled; } /// /// Clones the current instance (deep clone). /// /// Deep clone through method of helper class /// . /// Dictionary of all already clone objects. (Needed to avoid cycles.) /// The cloned object as . public override object Clone(IDictionary clonedObjects) { return new EngineBase(this, clonedObjects); } /// /// Calls /// of class . public virtual void Execute() { myRunning = true; myCanceled = false; ThreadPool.QueueUserWorkItem(new WaitCallback(Run), null); } /// /// Calls /// of class . public virtual void ExecuteSteps(int steps) { myRunning = true; myCanceled = false; ThreadPool.QueueUserWorkItem(new WaitCallback(Run), steps); } /// /// Calls /// of class . public void ExecuteStep() { ExecuteSteps(1); } /// /// Sets the protected flag myCanceled to true. public virtual void Abort() { myCanceled = true; } /// /// Sets myCanceled and myRunning to false. The global scope is cleared, /// the execution time is reseted, the execution stack is cleared and a new /// with the initial operator is added.
/// Calls .
public virtual void Reset() { myCanceled = false; myRunning = false; GlobalScope.Clear(); ExecutionTime = new TimeSpan(); myExecutionStack.Clear(); if (OperatorGraph.InitialOperator != null) myExecutionStack.Push(new AtomicOperation(OperatorGraph.InitialOperator, GlobalScope)); OnInitialized(); } private void Run(object state) { if (state == null) Run(); else RunSteps((int)state); myRunning = false; OnFinished(); } private void Run() { DateTime start = DateTime.Now; DateTime end; while ((!Canceled) && (!Terminated)) { ProcessNextOperation(); end = DateTime.Now; ExecutionTime += end - start; start = end; } ExecutionTime += DateTime.Now - start; } private void RunSteps(int steps) { DateTime start = DateTime.Now; DateTime end; int step = 0; while ((!Canceled) && (!Terminated) && (step < steps)) { ProcessNextOperation(); step++; end = DateTime.Now; ExecutionTime += end - start; start = end; } ExecutionTime += DateTime.Now - start; } /// /// Performs the next operation. /// protected abstract void ProcessNextOperation(); /// /// Occurs when the current instance is initialized. /// public event EventHandler Initialized; /// /// Fires a new Initialized event. /// protected virtual void OnInitialized() { if (Initialized != null) Initialized(this, new EventArgs()); } /// /// Occurs when an operation is executed. /// public event EventHandler OperationExecuted; /// /// Fires a new OperationExecuted event. /// /// The operation that has been executed. protected virtual void OnOperationExecuted(IOperation operation) { if (OperationExecuted != null) OperationExecuted(this, new OperationEventArgs(operation)); } /// /// Occurs when an exception occured during the execution. /// public event EventHandler ExceptionOccurred; /// /// Aborts the execution and fires a new ExceptionOccurred event. /// /// The exception that was thrown. protected virtual void OnExceptionOccurred(Exception exception) { Abort(); if (ExceptionOccurred != null) ExceptionOccurred(this, new ExceptionEventArgs(exception)); } /// /// Occurs when the execution time changed. /// public event EventHandler ExecutionTimeChanged; /// /// Fires a new ExecutionTimeChanged event. /// protected virtual void OnExecutionTimeChanged() { if (ExecutionTimeChanged != null) ExecutionTimeChanged(this, new EventArgs()); } /// /// Occurs when the execution is finished. /// public event EventHandler Finished; /// /// Fires a new Finished event. /// protected virtual void OnFinished() { if (Finished != null) Finished(this, new EventArgs()); } #region Persistence Methods /// /// Saves the current instance as in the specified . /// /// Calls of base class .
/// A quick overview how the single elements of the current instance are saved: /// /// /// Operator graph: /// Saved as a child node with the tag name OperatorGraph. /// /// /// Global scope: /// Saved as a child node with the tag name GlobalScope. /// /// /// Execution stack: /// A child node is created with the tag name ExecutionStack. Beyond this child node /// all operations of the execution stack are saved as child nodes. /// /// /// Execution time: /// Saved as a child node with the tag name ExecutionTime, where the execution /// time is saved as string in the node's inner text. /// ///
/// The (tag)name of the . /// The where to save the data. /// The dictionary of all already persisted objects. (Needed to avoid cycles.) /// The saved . public override XmlNode GetXmlNode(string name, XmlDocument document, IDictionary persistedObjects) { XmlNode node = base.GetXmlNode(name, document, persistedObjects); node.AppendChild(PersistenceManager.Persist("OperatorGraph", OperatorGraph, document, persistedObjects)); node.AppendChild(PersistenceManager.Persist("GlobalScope", GlobalScope, document, persistedObjects)); XmlNode stackNode = document.CreateNode(XmlNodeType.Element, "ExecutionStack", null); IOperation[] operations = new IOperation[ExecutionStack.Count]; ExecutionStack.CopyTo(operations, 0); for (int i = 0; i < operations.Length; i++) stackNode.AppendChild(PersistenceManager.Persist(operations[i], document, persistedObjects)); node.AppendChild(stackNode); XmlNode timeNode = document.CreateNode(XmlNodeType.Element, "ExecutionTime", null); timeNode.InnerText = ExecutionTime.ToString(); node.AppendChild(timeNode); return node; } /// /// Loads the persisted instance from the specified . /// /// See to get information on how the instance must be saved.
/// Calls of base class .
/// The where the engine is saved. /// The dictionary of all already restored objects. /// (Needed to avoid cycles.) public override void Populate(XmlNode node, IDictionary restoredObjects) { base.Populate(node, restoredObjects); myOperatorGraph = (IOperatorGraph)PersistenceManager.Restore(node.SelectSingleNode("OperatorGraph"), restoredObjects); myGlobalScope = (IScope)PersistenceManager.Restore(node.SelectSingleNode("GlobalScope"), restoredObjects); XmlNode stackNode = node.SelectSingleNode("ExecutionStack"); for (int i = stackNode.ChildNodes.Count - 1; i >= 0; i--) myExecutionStack.Push((IOperation)PersistenceManager.Restore(stackNode.ChildNodes[i], restoredObjects)); XmlNode timeNode = node.SelectSingleNode("ExecutionTime"); myExecutionTime = TimeSpan.Parse(timeNode.InnerText); } #endregion } }