5

在一个项目中,我正在尝试使用正则表达式来区分各种类型的句子并将它们映射到处理这些句子的函数。

这些句子处理函数中的大多数都从句子本身获取参数,并由正则表达式中的捕获组解析出来。

例如:“我为 2 个 cookie 支付了 20 美元”与我的解析树(字典)中的一个正则表达式匹配。正则表达式将提取 20 美元作为组“价格”,将 2 匹配为组“金额”。目前我正在映射到正确的 Handler 函数并按如下方式调用它:

foreach(KeyValuePair<Regex, Type> pair in sentenceTypes)
{
    Match match = pair.Key.Match(text);
    if(match.Success)
    {
        IHandler handler = handlerFactory.CreateHandler(pair.Value);
        output = handler.Handle(match);
    }
}

简单处理程序类的示例。

public class NoteCookiePriceHandler
    {
        public string Handle(Match match)
        {
            double payment = Convert.ToDouble(match.Result("${payment}"));
            int amount = Convert.ToInt32(match.Result("${amount}"));

            double price = payment / amount;
            return "The price is $" + price;
        }
    }

当我意识到我实际上无法模拟 Match 对象或 Regex 时,我试图用 Moq 设置一些单元测试来提供帮助。仔细想想,设计似乎总体上有些缺陷,因为我依赖于正确解析命名组并在没有良好接口的情况下将其交给 Handler 类。

我正在寻找有关更有效设计的建议,以将参数正确传递给映射的处理程序函数/类,因为传递 Match 对象似乎有问题。

如果做不到这一点,任何帮助找出有效模拟正则表达式或匹配的方法将不胜感激,至少可以帮助我解决我的短期问题。它们都缺少默认构造函数,因此我很难让 Moq 创建它们的对象。

编辑:我最终通过为我的匹配组传递一个字符串字典来解决至少模拟问题,而不是(un-Moq-able)匹配对象本身。我对这个解决方案不是特别满意,所以仍然会受到建议。

foreach(KeyValuePair<Regex, Type> pair in sentenceTypes)
        {
            match = pair.Key.Match(text);
            if(match.Success)
            {
                IHandler handler= handlerFactory.CreateHandler(pair.Value);
                foreach (string groupName in pair.Key.GetGroupNames())
                {
                    matchGroups.Add(groupName, match.Groups[groupName].Value);
                }
                interpretation = handler.Handle(matchGroups);
4

1 回答 1

1

避免糟糕设计的一种方法是从良好设计的原则开始,而不是简单地从您希望解决的问题开始。这就是为什么测试驱动开发在改变代码质量方面如此强大的原因之一。这种思维方式确实存在于 TDD 之前,尽管名称为:按合同设计。请允许我演示一下:

您希望理想的处理程序是什么样的?这个怎么样:

interface IHandler {
    String handle();
}

执行:

public class NoteCookiePriceHandler : IHandler
{  
    private double payment;
    private int amount;

    public NoteCookiePriceHandler(double payment, int amount) {
        this.payment = payment;
        this.amount = amount;
    }

    public String handle() {
        return "The price is $" + payment / amount;
    }
}

现在从这个理想的设计开始,也许是对这个设计的测试。我们如何获得要发送给处理程序的句子的句子输入?好吧,计算机科学中的所有问题都可以通过另一层间接来解决。假设句子解析器不直接创建处理程序,而是使用工厂来创建:

interface HandlerFactory<T> where T: IHandler  {
    T getHandler(KeyValuePair<String, String> captureGroups);
}

然后,您可以为每个处理程序创建一个工厂,但很快您就会找到一种创建通用工厂的方法。例如,使用反射,您可以将捕获组名称与构造函数参数匹配。根据构造函数参数的数据类型,您可以自动让您的通用处理程序工厂将您的字符串转换为正确的数据类型。通过创建一些假处理程序并要求工厂使用一些键值对字符串输入来填充它们,这一切都很容易测试。

于 2013-08-10T20:10:05.370 回答