[11659] | 1 | using System;
|
---|
| 2 | using System.Collections.Generic;
|
---|
| 3 | using System.Diagnostics;
|
---|
| 4 | using System.Linq;
|
---|
| 5 | using System.Text;
|
---|
| 6 | using System.Text.RegularExpressions;
|
---|
[11847] | 7 | using HeuristicLab.Encodings.SymbolicExpressionTreeEncoding;
|
---|
[11659] | 8 |
|
---|
| 9 | namespace HeuristicLab.Problems.GrammaticalOptimization {
|
---|
| 10 | // 4-bit even parity
|
---|
[11847] | 11 | public class EvenParityProblem : ISymbolicExpressionTreeProblem {
|
---|
[11659] | 12 | // + == OR
|
---|
| 13 | // * == AND
|
---|
| 14 | private const string grammarString = @"
|
---|
| 15 | G(S):
|
---|
[11770] | 16 | S -> a | b | c | d | a*S | b*S | c*S | d*S | a+S | b+S | c+S | d+S | !S | (S)
|
---|
[11659] | 17 | ";
|
---|
[11847] | 18 | // A = AND, O = OR, N = NOT, C = Clause
|
---|
| 19 | private const string hlGrammarString = @"
|
---|
| 20 | G(E):
|
---|
| 21 | E -> A | O | N | C | a | b | c | d
|
---|
| 22 | A -> EE | EEE
|
---|
| 23 | O -> EE | EEE
|
---|
| 24 | N -> E
|
---|
| 25 | C -> E
|
---|
| 26 | ";
|
---|
[11659] | 27 |
|
---|
| 28 | private readonly IGrammar grammar;
|
---|
[12099] | 29 | public string Name { get { return "EvenParity"; } }
|
---|
| 30 |
|
---|
[11659] | 31 | public EvenParityProblem() {
|
---|
[11770] | 32 | this.grammar = new Grammar(grammarString);
|
---|
[11857] | 33 | this.TreeBasedGPGrammar = new Grammar(hlGrammarString);
|
---|
[11659] | 34 | }
|
---|
| 35 |
|
---|
[11732] | 36 | public double BestKnownQuality(int maxLen) {
|
---|
[11659] | 37 | // for now only an upper bound is returned, ideally all fitness cases are predicted correctly
|
---|
[11847] | 38 | return 16;
|
---|
[11659] | 39 | }
|
---|
| 40 |
|
---|
| 41 | public IGrammar Grammar {
|
---|
| 42 | get { return grammar; }
|
---|
| 43 | }
|
---|
| 44 |
|
---|
| 45 | public double Evaluate(string sentence) {
|
---|
[11847] | 46 | var interpreter = new ExpressionInterpreter(); // for concurrent evaluation
|
---|
[11659] | 47 | var vars = new bool[4];
|
---|
| 48 | var nCorrect = 0;
|
---|
| 49 | for (int b0 = 0; b0 <= 1; b0++)
|
---|
| 50 | for (int b1 = 0; b1 <= 1; b1++)
|
---|
| 51 | for (int b2 = 0; b2 <= 1; b2++)
|
---|
| 52 | for (int b3 = 0; b3 <= 1; b3++) {
|
---|
| 53 | vars[0] = b0 > 0;
|
---|
| 54 | vars[1] = b1 > 0;
|
---|
| 55 | vars[2] = b2 > 0;
|
---|
| 56 | vars[3] = b3 > 0;
|
---|
| 57 |
|
---|
| 58 | var pred = interpreter.Interpret(sentence, vars);
|
---|
| 59 | var target = (b0 > 0) ^ (b1 > 0) ^ (b2 > 0) ^ (b3 > 0);
|
---|
| 60 | if (pred == target) nCorrect++;
|
---|
| 61 | }
|
---|
| 62 | return nCorrect;
|
---|
| 63 | }
|
---|
[11727] | 64 |
|
---|
[12893] | 65 | public bool IsOptimalPhrase(string phrase) {
|
---|
| 66 | throw new NotImplementedException();
|
---|
| 67 | }
|
---|
| 68 |
|
---|
[11832] | 69 | public string CanonicalRepresentation(string phrase) {
|
---|
[11770] | 70 | throw new NotImplementedException();
|
---|
[11832] | 71 | return phrase;
|
---|
[11727] | 72 | }
|
---|
[11832] | 73 |
|
---|
[12893] | 74 | public IEnumerable<Feature> GetFeatures(string phrase) {
|
---|
| 75 | return new[] { new Feature(phrase, 1.0) };
|
---|
[11832] | 76 | }
|
---|
[11847] | 77 |
|
---|
[11857] | 78 | public IGrammar TreeBasedGPGrammar { get; private set; }
|
---|
[11847] | 79 | public string ConvertTreeToSentence(ISymbolicExpressionTree tree) {
|
---|
| 80 | var sb = new StringBuilder();
|
---|
| 81 |
|
---|
| 82 | TreeToSentence(tree.Root.GetSubtree(0).GetSubtree(0), sb);
|
---|
| 83 |
|
---|
| 84 | return sb.ToString();
|
---|
| 85 | }
|
---|
| 86 |
|
---|
| 87 | private void TreeToSentence(ISymbolicExpressionTreeNode treeNode, StringBuilder sb) {
|
---|
| 88 | if (treeNode.SubtreeCount == 0) {
|
---|
| 89 | // terminal
|
---|
| 90 | sb.Append(treeNode.Symbol.Name);
|
---|
| 91 | } else {
|
---|
| 92 | switch (treeNode.Symbol.Name) {
|
---|
| 93 | case "O": {
|
---|
| 94 | sb.Append("(");
|
---|
| 95 | TreeToSentence(treeNode.Subtrees.First(), sb);
|
---|
| 96 | foreach (var subTree in treeNode.Subtrees.Skip(1)) {
|
---|
| 97 | sb.Append("+");
|
---|
| 98 | TreeToSentence(subTree, sb);
|
---|
| 99 | }
|
---|
| 100 | sb.Append(")");
|
---|
| 101 | break;
|
---|
| 102 | }
|
---|
| 103 | case "A": {
|
---|
| 104 | TreeToSentence(treeNode.Subtrees.First(), sb);
|
---|
| 105 | foreach (var subTree in treeNode.Subtrees.Skip(1)) {
|
---|
| 106 | sb.Append("*");
|
---|
| 107 | TreeToSentence(subTree, sb);
|
---|
| 108 | }
|
---|
| 109 | break;
|
---|
| 110 | }
|
---|
| 111 | case "N": {
|
---|
| 112 | Debug.Assert(treeNode.SubtreeCount == 1);
|
---|
| 113 | sb.Append("!(");
|
---|
| 114 | TreeToSentence(treeNode.Subtrees.Single(), sb);
|
---|
| 115 | sb.Append(")");
|
---|
| 116 | break;
|
---|
| 117 | }
|
---|
| 118 | case "C": {
|
---|
| 119 | Debug.Assert(treeNode.SubtreeCount == 1);
|
---|
| 120 | sb.Append("(");
|
---|
| 121 | TreeToSentence(treeNode.Subtrees.Single(), sb);
|
---|
| 122 | sb.Append(")");
|
---|
| 123 | break;
|
---|
| 124 | }
|
---|
| 125 | default: {
|
---|
| 126 | Debug.Assert(treeNode.SubtreeCount == 1);
|
---|
| 127 | TreeToSentence(treeNode.Subtrees.Single(), sb);
|
---|
| 128 | break;
|
---|
| 129 | }
|
---|
| 130 | }
|
---|
| 131 | }
|
---|
| 132 | }
|
---|
[11659] | 133 | }
|
---|
| 134 | }
|
---|