аккуратная реализация условий IF на языке сценариев - PullRequest
0 голосов
/ 03 июля 2018

У меня есть небольшой вспомогательный класс, который генерирует скрипт, который позже используется в моем коде, а именно:

public class ScriptBuilder
{
    public string Script { get; set; }
    public ScriptBuilder NewLine(uint numberOfLines = 1)
    {
        if (numberOfLines == 0)
        {
            return this;
        }
        else
        {
            for (int i = 1; i <= numberOfLines; ++i)
            {
                Script += Environment.NewLine;
            }
            return this;
        }
    }
    public ScriptBuilder WriteLine(string str = "")
    {
        if (str != "")
        {
            Script += str;
            NewLine();
        }
        return this;
    }
    public ScriptBuilder(string line = "")
    {
        Script = line;
        if (line != "")
        {
            NewLine();
        }
    }
    public ScriptBuilder setLong(string longName, long x)
    {
        WriteLine("int " + longName + " " + x.ToString(System.Globalization.CultureInfo.InvariantCulture));
        return this;
    }

    // + other set functions with different parameters/numbers of parameters
}

// ScriptBuilder is used like this :

ScriptBuilder scriptStringBuilder = new ScriptBuilder();
scriptStringBuilder
    .WriteLine($"/!HEADSTART")
    .WriteLine($"/! TYPE = {scriptType}")
    .WriteLine($"/! NAME = {name}")
    .WriteLine($"/! DESCRIPTION  = {description}")
    .WriteLine($"/!HEADEND")
    /* the header is done now */
    .NewLine(2);

Это довольно простой метод создания цепочек. Я хотел бы реализовать IF и IF ELSE IF ELSE на этом языке сценариев, но я не вижу действительно изящного способа сделать это.

Для IF я придумал функцию-член:

public ScriptBuilder IF(bool condition, ScriptBuilder res)
{
    if (condition)
    {
        return res;
    }
    else
    {
        return this;
    }
}

, который можно использовать как:

ScriptBuilder.IF(condition,
    scriptStringBuilder
        .setThis(...)
        .setThat(...)
    ;
    )

но я не удовлетворен, потому что

  • scriptStringBuilder.setThis(...).setThat(...); уже бы вступил в силу
  • Я должен написать scriptStringBuilder, чтобы использовать его. Хотя нет подходящего представления о IF ELSE IF ELSE.

Я мог бы использовать делегатов, например:

public delegate ScriptBuilder ScriptBuilderFunction(params object[] Parameters);
public ScriptBuilder IF(bool condition, ScriptBuilderFunction func)
{
    // ...
}

но я даже не вижу, как это реализовать ...

В идеале я хотел бы написать:

scriptStringBuilder.
    .setThis(...)
    .setThat(...)
    .IF(condition)
    .THEN()
        .setThis(...)
        .doThat(...)
    .ELSEIF(othercondition)
        .makeThis(...)
        .doThat(...)
    .ENDIF()
    .setThatNow(...)
    ;

1 Ответ

0 голосов
/ 03 июля 2018

Это очень грубый набросок того, как вы могли бы реализовать это:

class Program
{
    static void Main(string[] args)
    {
        var number = 999;

        ScriptBuilder scriptStringBuilder = new ScriptBuilder();
        var text = scriptStringBuilder
            .WriteLine($"/!HEADSTART")
            .WriteLine($"/! TYPE = abc")
            .WriteLine($"/! NAME = name")
            .WriteLine($"/!HEADEND")
            .NewLine(2)
            .IfCondition(number != 999, ifCondition => {
                ifCondition.NewLine(1);
                ifCondition.WriteLine("SUCCESS");
            }, elseCondition => {
                elseCondition.NewLine(1);
                elseCondition.WriteLine("FAIL");
            }, 
            elseIf1 => elseIf1.ElseIfCondition(number > 1, h1 => h1.WriteLine("NUMBER IS BIGGER THAN 1")),
            elseIf2 => elseIf2.ElseIfCondition(number > 2, h2 => h2.WriteLine("NUMBER IS BIGGER THAN 2")))
            .Build();

        Console.WriteLine(text);

        Console.ReadKey();
    }
}

public class ScriptBuilder : IElseIfConditionable
{
    private string _script;

    public ScriptBuilder NewLine(uint numberOfLines = 1)
    {
        if (numberOfLines == 0)
        {
            return this;
        }
        else
        {
            for (int i = 1; i <= numberOfLines; ++i)
            {
                _script += Environment.NewLine;
            }
            return this;
        }
    }
    public ScriptBuilder WriteLine(string str = "")
    {
        if (str != "")
        {
            _script += str;
            NewLine();
        }
        return this;
    }
    public ScriptBuilder(string line = "")
    {
        _script = line;
        if (line != "")
        {
            NewLine();
        }
    }
    public ScriptBuilder SetLong(string longName, long x)
    {
        WriteLine("int " + longName + " " + x.ToString(System.Globalization.CultureInfo.InvariantCulture));
        return this;
    }

    public string Build()
    {
        return _script;
    }

    public ScriptBuilder IfCondition(bool condition, Action<ScriptBuilder> trueCondition, Action<ScriptBuilder> falseCondition, params Func<IElseIfConditionable, Tuple<ScriptBuilder, bool>>[] elseIfs)
    {
        if (condition)
        {
            trueCondition(this);
            return this;
        }

        foreach (var elseIf in elseIfs)
        {
            if (elseIf(this).Item2)
            {
                return this;
            }
        }

        if (!condition)
        {
            falseCondition(this);
        }

        return this;
    }

    public Tuple<ScriptBuilder, bool> ElseIfCondition(bool condition, Action<ScriptBuilder> trueCondition)
    {
        if (condition)
        {
            trueCondition(this);
        }

        return Tuple.Create(this, condition);
    }
}

public interface IElseIfConditionable
{
    Tuple<ScriptBuilder, bool> ElseIfCondition(bool condition, Action<ScriptBuilder> trueCondition);
}

Это эквивалентно:

var s = new ScriptBuilder();
if (number != 999)
{
    s.NewLine(1);
    s.WriteLine("SUCCESS");
}
else if (number > 1)
{
    s.WriteLine("NUMBER IS BIGGER THAN 1");
}
else if (number > 2)
{
    s.WriteLine("NUMBER IS BIGGER THAN 2");
}
else
{
     s.NewLine(1);
     s.WriteLine("FAIL");
}
...