» The Interpreter pattern supports the interpretation of instructions written in a language or notation defined for a specific purpose. The notation is precise and can be defined in a form of grammar; » A common form of grammar is to make use of a set of terms. Terms are either defined as terminals (these are final terms) or as nonterminals (these comprise other terms); » The Interpreter Pattern matches supplied input (containing instructions and input data) against a predefined grammar to create an object structure; this process is referred to as parsing; » Once the object structure is created, these objects can be used to perform operations on the input and generate a resulting output; the operations are performed in the context of some input and output; » The Interpreter Pattern is not designed for large grammars, and can suffer performance problems with more complex grammars.
Use When:
» You have a grammar to be interpreted and the grammar can be translated into a set of classes; » you have a grammar to be interpreted and the grammar is not too large; » you have a grammar to be interpreted and efficiency is not critical.
//Define the Element Base Class and Associated Child Clases.
//Define the Element class which contains all the core functions to be used in this pattern. //All other classes will derive from this class.
//The input data to this class is provided by the Context class, and is declared in a predefined grammatical syntax. //The Parse() and ParseNumber() methods work on the input data to generate an object structure of objects derived from the Element class (object). //These include Lab, Test, MidTerm and Exam objects. //Once the input data has been organised into the object structure, we can execute "interpreter" methods against the data to generate any outputs required. //The ParseResults() method calculates ("interprets") student course results based on predefined weighting for each element of the course. public classElement { public int Weight { get; set; } public byte Level { get; set; } publicElement Next { get; set; } publicElement Child { get; set; } public static int SumLab { get; set; } public static int SumTest { get; set; }
public static int[] MValues; public static int MCounter; private static byte _mLevel = 1;
public virtual string Display() { return (string.Concat(Weight, "%")); }
private int ParseNumber(Context context) { int posX = context.Input.IndexOf('|'); int value = int.Parse(context.Input.Substring(1, posX - 1)); context.Input = context.Input.Substring(posX + 1);
return value; }
private void ParseResults(Context context) { if (GetType() == typeof(Lab) || GetType() == typeof(Test)) { context.Output += MValues[MCounter] * Weight; MCounter++; } else if ((this is MidTerm || this is Exam) && Child == null) { context.Output += MValues[MCounter] * Weight; MCounter++; }
if (Child != null) { Child.ParseResults(context); }
if (Next != null) { Next.ParseResults(context); } } }
//The Course class (object) represents the first element in the object structure. //This class inherits from the Element class, and therefore inherits all the methods defined in the Element class. public classCourse : Element { public string Name { get; set; }
public Course(Context context) { Name = context.Input.Substring(0, 6); context.Input = context.Input.Substring(7); }
public override string Display() { return Name; }
public override string ToString() { return"Course"; } }
//The Lab class (object) represents another element in the object structure. //This class inherits from the Element class, and therefore inherits all the methods defined in the Element class. public classLab : Element { public override string ToString() { return"Lab"; } }
//The Test class (object) represents another element in the object structure. //This class inherits from the Element class, and therefore inherits all the methods defined in the Element class. public classTest : Element { public override string ToString() { return"Test"; } }
//The MidTerm class (object) represents another element in the object structure. //This class inherits from the Element class, and therefore inherits all the methods defined in the Element class. public classMidTerm : Element { public override string ToString() { return"MidTerm"; } }
//The Exam class (object) represents another element in the object structure. //This class inherits from the Element class, and therefore inherits all the methods defined in the Element class. public classExam : Element { public override string ToString() { return"Exam"; } }
//The Context class (object) is simple class that stores that input data to, and output data from the Element class (object). public class Context { public string Input { get; set; } public double Output { get; set; }
public Context(string c) { Input = c; Output = 0; } }