c# - 哪些后编译器(或其他)选项可用于在结构之间重用功能?

标签 c# .net t4 postsharp

假设我创建了一个名为 Percentstruct。从逻辑上讲,它作为一个结构是有意义的,因为它代表一个值,并且在使用时应该按值传递。该结构还包含一些属性和静态隐式运算符。 Percent 可以处理任何类型的百分比,包括超过 100% 的百分比。

现在,假设我希望能够在只关心 0-100% 之间的百分比的应用程序中使用 Percent。我决定创建一个名为 ValidPercent 的新结构,它实际上与 Percent 相同,除了进行额外检查以确保 ValidPercent 永远不会包含高于 100% 或低于 0% 的值.

ValidPercent 似乎很适合继承,但结构无法利用继承。此外,能够将 ValidPercent 向下转换为 Percent 的值(value)很小。

是否有任何可用的工具可以让我定义 Percent 和 ValidPercent,而无需重复每个工具中使用的大部分代码?

我对 PostSharp 和 t4 模板做了一些粗略的研究,但我无法确定它们是否可以用于此目的。他们可以吗,或者是否有其他可用的工具可以?

最佳答案

T4 可用于此。这是最好的选择吗?这取决于您的场景(我假设您发布了真实场景的简化版本)。

无论如何,可以定义一个生成多个变体的模板。使用部分类和方法,您可以将特定行为注入(inject)生成的代码(例如验证)。

您可以在这里找到完整的源代码:https://github.com/mrange/CodeStack/tree/master/q18861246/TestProject

我使用 VS2013,但这在 VS2008+ 中可以正常工作。

我定义了一个 T4 模板:

<#
    // The model defines *what* we like generated
    var model = new []
    {
        "ValidPercent"      ,
        "Percent"           ,
    };
#>

namespace TestProject
{
<#
    // The "View" defines *how* the model is transformed into code
    foreach (var cls in model)
    {
#>

    partial struct <#=cls#>
    {
        // Partial struct/class are great with T4 or any code-generation tool

        decimal m_value;

        // Partial methods are great to inject customized behavior into the generated code skeleton
        static partial void Partial_ValidateValue (decimal value);

        public <#=cls#> (decimal value)
        {
            Partial_ValidateValue (value);
            m_value = value;
        }

        public decimal Value 
        {
            get
            {
                return m_value;
            }
            set
            {
                Partial_ValidateValue (value);
                m_value = value;
            }
        }

        public override string ToString ()
        {
            return Value + "%";
        }

    }
<#
    }
#>

}

为了编写可维护的元程序(我的首选术语),将模型即我们希望生成的与 View 分开是一种很好的做法模型如何转换为代码

在这种情况下,模型非常简单:

// The model defines *what* we like generated
var model = new []
{
    "ValidPercent"      ,
    "Percent"           ,
};

View 基本上只是迭代生成代码的模型。 T4 基本上就像 ASP/PHP。

<#
    // The "View" defines *how* the model is transformed into code
    foreach (var cls in model)
    {
#>
...

为了能够注入(inject)验证行为,我在生成的代码中插入了一个扩展点:

// Partial methods are great to inject customized behavior into the generated code skeleton
static partial void Partial_ValidateValue (decimal value);

部分方法基本上像事件一样工作,但它们在编译时 Hook 。在分配 m_value 之前调用 Partial_ValidateValue 以确保支持任何类不变量。

为了注入(inject)验证行为,我在单独的文件中定义了类 ValidPercent 的另一部分:

partial struct ValidPercent 
{
    public static implicit operator Percent(ValidPercent vp)
    {
        return new Percent (vp.Value);
    }

    static partial void Partial_ValidateValue(decimal value)
    {
        if (value < 0M || value > 100M)
        {
            throw new ArgumentException ("value", "value is expected to be in the range 0..100");
        }
    }
}

该运算符只是一个方便的运算符,允许从 ValidPercent ==> Percent 进行隐式转换(这始终是安全的)。 Partial_ValidateValue 进行实际验证。

在考虑 T4 是否适合您时,这应该可以为您提供一些起点。

希望对你有帮助

关于c# - 哪些后编译器(或其他)选项可用于在结构之间重用功能?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/18861246/

相关文章:

c# - 如何将数据从 PC 发送到手机 c#

c# - 如何找到 ResourceManager 中可用的资源文化?

.net - 使用 Web 服务公开 .NET DAL 是否会增加安全性?

c# - 生成的 C# 代码中的 T4 缩进

asp.net-mvc - MVC脚手架: get current controller name and area name in T4 template

c# - 是否有属性/方法可以确定 TcpListener 当前是否正在监听?

c# - 如何覆盖 newtonsoft json 中的 "Required.Always"

c# - 从 AD 获取经理的员工

.net - 如何向 .NET 应用程序添加安全的 FTP 功能

c# - T4 获取解决方案的当前工作目录