2009-08-09 3 views
12

Quindi voglio essere in grado di analizzare e valutare "espressioni di dadi" in C#. Un'espressione di dadi è definita in questo modo:Espressione di dadi di analisi (ad esempio 3d6 + 5) in C#: da dove iniziare?

<expr> := <expr> + <expr> 
      | <expr> - <expr> 
      | [<number>]d(<number>|%) 
      | <number> 
<number> := positive integer 

Così ad es. d6+20-2d3 sarebbe stato permesso, e dovrebbe valutare come

rand.Next(1, 7) + 20 - (rand.Next(1, 4) + rand.Next(1, 4)) 

anche d% dovrebbe essere equivalente a d100.

So che potrei mettere insieme qualche soluzione, ma so anche che questo sembra un tipico problema di tipo informatico, quindi ci deve essere una soluzione super elegante che dovrei esaminare.

mi piacerebbe il risultato della mia analisi di avere queste capacità:

  • dovrei essere in grado di emettere una forma normalizzata dell'espressione; Prima sto pensando ai dadi, ordinati per dimensione dei dadi e sempre con un prefisso. Quindi ad es. il campione precedente diventerebbe 1d6-2d3+20. Anche le istanze di d% diventerebbero d100 nel formato normalizzato.
  • Dovrei essere in grado di valutare l'espressione a volontà, facendo ruotare diversi numeri casuali ogni volta.
  • Dovrei essere in grado di valutare l'espressione con tutti i rulli del dado massimizzati, quindi ad es. l'esempio sopra darebbe (in modo deterministico) 1*6+20+2*3 = 32.

So che questo è esattamente il tipo di cosa che Haskell e probabilmente altri linguaggi di tipo funzionale sarebbero fantastici, ma mi piacerebbe rimanere in C# se possibile.

I miei pensieri iniziali tendono verso la ricorsione, le liste e forse qualche LINQ, ma ancora una volta, se provassi senza alcune indicazioni da persone che sanno cose, sono sicuro che finirebbe per essere un disastro inelegante.

Un'altra tattica che potrebbe funzionare sarebbe una sostituzione di stringa basata su espressioni regex per trasformare le espressioni di dadi nelle chiamate rand.Next e quindi la valutazione o la compilazione al volo ... funzionerebbe davvero? Come evitare di creare ogni volta un nuovo oggetto rand?

+0

di random.Next è comprensivo, ed è secondo parametro è esclusivo. http://msdn.microsoft.com/en-us/library/2dx6wyd4%28VS.95%29.aspx – Domenic

+0

abbastanza giusto :) – annakata

+0

Immagino tu intenda: rand.Next (1, 7) + 20 - (rand. Successivo (1, 4) + rand.Next (1, 4)) –

risposta

9

Ecco quello che alla fine si avvicinò con: primo parametro

using System; 
using System.Collections.Generic; 
using System.Linq; 
using System.Text.RegularExpressions; 

public enum DiceExpressionOptions 
{ 
    None, 
    SimplifyStringValue 
} 
public class DiceExpression 
{ 
    /* <expr> := <expr> + <expr> 
    *   | <expr> - <expr> 
    *   | [<number>]d(<number>|%) 
    *   | <number> 
    * <number> := positive integer 
    * */ 
    private static readonly Regex numberToken = new Regex("^[0-9]+$"); 
    private static readonly Regex diceRollToken = new Regex("^([0-9]*)d([0-9]+|%)$"); 

    public static readonly DiceExpression Zero = new DiceExpression("0"); 

    private List<KeyValuePair<int, IDiceExpressionNode>> nodes = new List<KeyValuePair<int, IDiceExpressionNode>>(); 

    public DiceExpression(string expression) 
     : this(expression, DiceExpressionOptions.None) 
    { } 
    public DiceExpression(string expression, DiceExpressionOptions options) 
    { 
     // A well-formed dice expression's tokens will be either +, -, an integer, or XdY. 
     var tokens = expression.Replace("+", " + ").Replace("-", " - ").Split(' ', StringSplitOptions.RemoveEmptyEntries); 

     // Blank dice expressions end up being DiceExpression.Zero. 
     if (!tokens.Any()) 
     { 
      tokens = new[] { "0" }; 
     } 

     // Since we parse tokens in operator-then-operand pairs, make sure the first token is an operand. 
     if (tokens[0] != "+" && tokens[0] != "-") 
     { 
      tokens = (new[] { "+" }).Concat(tokens).ToArray(); 
     } 

     // This is a precondition for the below parsing loop to make any sense. 
     if (tokens.Length % 2 != 0) 
     { 
      throw new ArgumentException("The given dice expression was not in an expected format: even after normalization, it contained an odd number of tokens."); 
     } 

     // Parse operator-then-operand pairs into this.nodes. 
     for (int tokenIndex = 0; tokenIndex < tokens.Length; tokenIndex += 2) 
     { 
      var token = tokens[tokenIndex]; 
      var nextToken = tokens[tokenIndex + 1]; 

      if (token != "+" && token != "-") 
      { 
       throw new ArgumentException("The given dice expression was not in an expected format."); 
      } 
      int multiplier = token == "+" ? +1 : -1; 

      if (DiceExpression.numberToken.IsMatch(nextToken)) 
      { 
       this.nodes.Add(new KeyValuePair<int, IDiceExpressionNode>(multiplier, new NumberNode(int.Parse(nextToken)))); 
      } 
      else if (DiceExpression.diceRollToken.IsMatch(nextToken)) 
      { 
       var match = DiceExpression.diceRollToken.Match(nextToken); 
       int numberOfDice = match.Groups[1].Value == string.Empty ? 1 : int.Parse(match.Groups[1].Value); 
       int diceType = match.Groups[2].Value == "%" ? 100 : int.Parse(match.Groups[2].Value); 
       this.nodes.Add(new KeyValuePair<int, IDiceExpressionNode>(multiplier, new DiceRollNode(numberOfDice, diceType))); 
      } 
      else 
      { 
       throw new ArgumentException("The given dice expression was not in an expected format: the non-operand token was neither a number nor a dice-roll expression."); 
      } 
     } 

     // Sort the nodes in an aesthetically-pleasing fashion. 
     var diceRollNodes = this.nodes.Where(pair => pair.Value.GetType() == typeof(DiceRollNode)) 
             .OrderByDescending(node => node.Key) 
             .ThenByDescending(node => ((DiceRollNode)node.Value).DiceType) 
             .ThenByDescending(node => ((DiceRollNode)node.Value).NumberOfDice); 
     var numberNodes = this.nodes.Where(pair => pair.Value.GetType() == typeof(NumberNode)) 
            .OrderByDescending(node => node.Key) 
            .ThenByDescending(node => node.Value.Evaluate()); 

     // If desired, merge all number nodes together, and merge dice nodes of the same type together. 
     if (options == DiceExpressionOptions.SimplifyStringValue) 
     { 
      int number = numberNodes.Sum(pair => pair.Key * pair.Value.Evaluate()); 
      var diceTypes = diceRollNodes.Select(node => ((DiceRollNode)node.Value).DiceType).Distinct(); 
      var normalizedDiceRollNodes = from type in diceTypes 
              let numDiceOfThisType = diceRollNodes.Where(node => ((DiceRollNode)node.Value).DiceType == type).Sum(node => node.Key * ((DiceRollNode)node.Value).NumberOfDice) 
              where numDiceOfThisType != 0 
              let multiplicand = numDiceOfThisType > 0 ? +1 : -1 
              let absNumDice = Math.Abs(numDiceOfThisType) 
              orderby multiplicand descending 
              orderby type descending 
              select new KeyValuePair<int, IDiceExpressionNode>(multiplicand, new DiceRollNode(absNumDice, type)); 

      this.nodes = (number == 0 ? normalizedDiceRollNodes 
             : normalizedDiceRollNodes.Concat(new[] { new KeyValuePair<int, IDiceExpressionNode>(number > 0 ? +1 : -1, new NumberNode(number)) })).ToList(); 
     } 
     // Otherwise, just put the dice-roll nodes first, then the number nodes. 
     else 
     { 
      this.nodes = diceRollNodes.Concat(numberNodes).ToList(); 
     } 
    } 

    public override string ToString() 
    { 
     string result = (this.nodes[0].Key == -1 ? "-" : string.Empty) + this.nodes[0].Value.ToString(); 
     foreach (var pair in this.nodes.Skip(1)) 
     { 
      result += pair.Key == +1 ? " + " : " − "; // NOTE: unicode minus sign, not hyphen-minus '-'. 
      result += pair.Value.ToString(); 
     } 
     return result; 
    } 
    public int Evaluate() 
    { 
     int result = 0; 
     foreach (var pair in this.nodes) 
     { 
      result += pair.Key * pair.Value.Evaluate(); 
     } 
     return result; 
    } 
    public decimal GetCalculatedAverage() 
    { 
     decimal result = 0; 
     foreach (var pair in this.nodes) 
     { 
      result += pair.Key * pair.Value.GetCalculatedAverage(); 
     } 
     return result; 
    } 

    private interface IDiceExpressionNode 
    { 
     int Evaluate(); 
     decimal GetCalculatedAverage(); 
    } 
    private class NumberNode : IDiceExpressionNode 
    { 
     private int theNumber; 
     public NumberNode(int theNumber) 
     { 
      this.theNumber = theNumber; 
     } 
     public int Evaluate() 
     { 
      return this.theNumber; 
     } 

     public decimal GetCalculatedAverage() 
     { 
      return this.theNumber; 
     } 
     public override string ToString() 
     { 
      return this.theNumber.ToString(); 
     } 
    } 
    private class DiceRollNode : IDiceExpressionNode 
    { 
     private static readonly Random roller = new Random(); 

     private int numberOfDice; 
     private int diceType; 
     public DiceRollNode(int numberOfDice, int diceType) 
     { 
      this.numberOfDice = numberOfDice; 
      this.diceType = diceType; 
     } 

     public int Evaluate() 
     { 
      int total = 0; 
      for (int i = 0; i < this.numberOfDice; ++i) 
      { 
       total += DiceRollNode.roller.Next(1, this.diceType + 1); 
      } 
      return total; 
     } 

     public decimal GetCalculatedAverage() 
     { 
      return this.numberOfDice * ((this.diceType + 1.0m)/2.0m); 
     } 

     public override string ToString() 
     { 
      return string.Format("{0}d{1}", this.numberOfDice, this.diceType); 
     } 

     public int NumberOfDice 
     { 
      get { return this.numberOfDice; } 
     } 
     public int DiceType 
     { 
      get { return this.diceType; } 
     } 
    } 
} 
5

potresti usare la tua grammatica in un compilatore-compilatore (qualcosa come Yacc) per C# (come antlr) o semplicemente iniziare a scrivere il tuo recursive descent parser.

Poi si costruisce una in memoria struttura dati (un albero, se volete operazioni arbitrarie matematiche diverse +) che is Visitable so you need to write a couple of visitors:

  • RollVisitor: init un seme rand poi la visita ogni nodo, risultato
  • accumulando
  • GetMaxVisitor: somma il limite superiore di ogni dado
  • altri visitatori? (Come ad esempio PrettyPrintVisitor, RollTwiceVisitor, etc etc)

penso che un visitabile-albero è una soluzione degna qui.

+0

Anche se per essere onesti, questo mi sembra eccessivo. –

+0

@Greg: è un progetto standard per un albero di analisi in modalità OO ... perché pensi che sia eccessivo? preferisci una singola riga piena di espressioni regolari? – dfa

+0

Non ho analizzato la grammatica fornita, ma sembra abbastanza piccolo che una vera soluzione codegen potrebbe essere meno semplice di quanto lo spazio del problema richieda. Se avessi il mio testo di riferimento con me per un aggiornamento della memoria, sarei tentato di disegnare gli automi appropriati sul posto. –