c# - 如何在 C# 中编写一个很好的奇特的重复模板模式 (CRTP)

标签 c# crtp

前阵子我想创建我自己的数据映射器,它比普通的 ORM 简单得多。在这样做的过程中,我发现需要访问我的基类中继承类的类型信息。我的第一个想法是反射,但它太慢了(如果你使用反射,请查看 Fasterflect 因为它“几乎”消除了反射的性能问题)。

所以我求助于一个解决方案,后来我发现它有自己的名字:The Curiously Recurring Template Pattern。这主要解决了我的问题,但学习如何正确实现此模式有点挑战。我必须解决的两个主要问题是:

1) 如何让我的消费代码与我的通用对象一起工作,而不需要知道创建对象时使用的通用参数?

2) 如何在 C# 中继承静态字段?

具有挑战性的部分实际上是解决问题。一旦我意识到我需要做什么,解决这些问题就很容易了。如果您发现自己需要 CRTP,您可能会发现自己需要回答这些问题……它们似乎是齐头并进的。

最佳答案

在不知道泛型参数类型的情况下使用泛型

当使用 CRTP 时,最好有一个非泛型基类(如果可能的话抽象,但这不是太重要)你的“基”泛型类继承自。然后,您可以在非泛型基类上创建抽象(或虚拟)函数,并允许使用代码来处理您的对象,而无需知道泛型参数。例如:

abstract class NonGenBase
{
    public abstract void Foo();
}

class GenBase<T>: NonGenBase
{
    public override void Foo()
    {
        // Do something
    }
}

现在使用不知道 T 应该是什么的代码仍然可以通过将对象视为基类的实例来调用对象上的 Foo() 过程。

如何解决静态字段继承问题

当使用 CRTP 解决问题时,在继承类中提供对静态字段的访问通常是有益的。问题是 C# 不允许继承类访问那些静态字段,除非通过类型名称……在这种情况下这似乎常常违背了目的。您可能想不出我在说什么的一个清晰的例子,解释一个超出了这个答案的范围,但解决方案很简单,所以只要把它藏在你的知识库中,当你发现需要它的时候你会很高兴它在那里:)

class GenBase<T>: NonGenBase
{
    static object _someResource;

    protected object SomeResource { get { return _someResource; } }
}

这“模拟”了静态字段的继承。但是请记住,泛型类上的静态字段并没有涵盖所有泛型实现。每个通用实现都有自己的静态字段实例。如果您想要一个可用于所有实现的静态字段,那么您只需将它添加到您的非泛型基类中。

关于c# - 如何在 C# 中编写一个很好的奇特的重复模板模式 (CRTP),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/10939907/

相关文章:

c# - 多表连接优化性能

c# - 用于在运行时从接口(interface)生成具体类型的开源库?

c++ - 为什么自动返回类型推导适用于未完全定义的类型?

c++ - CRTP 派生类方法的返回类型的转发

c++ - CRTP 自动注册工厂(静态初始化顺序失败)

c# - 如何编写一个将不同类型作为参数的泛型方法?

c# - 用于服务应用程序的 Google oAuth 2.0(JWT token 请求)

c# - 检查对象类中的电话号码

C++ CRTP 问题,MSVC 错误 C2039