如何在C#中编写一个好奇怪的循环模板模式(CRTP)

前端之家收集整理的这篇文章主要介绍了如何在C#中编写一个好奇怪的循环模板模式(CRTP)前端之家小编觉得挺不错的,现在分享给大家,也给大家做个参考。
一段时间后,我想创建自己的数据映射器,这将比您的平均ORM简单得多.在这样做时,我发现需要访问我的基类中继承类的类型信息.我的第一个想法是反思,但是太慢了(如果你使用反射,看看 Fasterflect,因为它几乎“消除了反射的性能问题).

所以我转向一个解决方案,我后来发现有自己的名字:好奇的循环模板模式.这主要解决了我的问题,但是学习如何正确地实现这种模式有点具有挑战性.我必须解决的两个主要问题是:

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#相关文章