c# - 是否有一种(优雅的)解决方案可以在方法中进一步约束泛型类型参数?

标签 c# generics type-constraints

我有一个通用基类 Foo<T>来自哪些类(class)Bar<U>Bat<T>派生。 U源自 T . Bat 和 Bar 是相似的实现,仅在少数地方不同,其中类型为 U 的值必须以不同的方式处理。

Foo , 我有一个工厂方法 Create接受类型为 T 的参数并且应该创建一个 BarBat目的。 它看起来大致是这样的:

public static IFoo<T> Create(T input) {
  if (input.TypeIdentifier == Types.Bar) {// exemplary type check
    // input is of or derives from `U`
    // return a Bar<U>
  } else 
    return new Bat(input);
}

// usage:
U myU = new ClassThatDerivesFromU();
T myT = new ClassThatDerivesFromT(CouldBe.Of(Type.U));
var myFoo1 = Create(myU); // of type IFoo<U>
var myFoo2 = Create(myT); // of type IFoo<T>

T不是 U ,我无法实例化 Bar对象。

一个可能的解决方案是这样的:

public static U To<T, U>(T input) where U : T {
  return input as U;
}

// to create Bar:
new Bar(To<T, U>(input));

然而,这在我看来是个 hack,不能与结构一起使用(在这种情况下,U 由于继承而不能成为结构,但我有另一种情况,我想根据 if T 调用方法。例如 structclass

在 C++ 中,可以通过提供 Create 的多个重载来解决 (iirc) 这样的场景具有不同类型约束的方法,编译器将检查类型 T并选择正确的方法(使用 UT 作为类型约束)。

我不知道 C# 中有类似的内置解决方案,但也许我可以使用一种优雅的解决方法? (反射是一个明显的答案,但不是一个选项)

最佳答案

是的,在使用接口(interface)时允许泛型变化。您可以将 IFoo 中的泛型类型参数声明为协变或逆变(取决于用法)。如果你想使用更派生的类型,那么类型 T必须是逆变的,并且 IFoo 可以声明如下:

interface IFoo<in T> { ... }

看看this MSDN page有关 C# 中泛型和变体的更多信息。

更新

一旦你有条件 IFoo<U>IFoo<T>如果U : T (例如,IFoo 的通用类型是逆变的)然后您可以在您的创建方法中自由安全地进行转换:

return (IFoo<T>)((object)new Bar<U>());

关于c# - 是否有一种(优雅的)解决方案可以在方法中进一步约束泛型类型参数?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13816390/

相关文章:

c# - 在单个字符串上运行多个 RegEx 模式

C# 具有多个函数重载的事件

c# - 如何向 Windows Phone 8.1 应用程序添加循环选择器?

c# - 动态对象 - 整洁的属性名称?

generics - Swift 通用类型变量应遵守协议(protocol)(类型类)

c# - 删除文件夹 C# .net Core

大于/小于 : <> are they class specific? 的 Java 语法

Swift泛型强制的误解

具有 "where constraint"定义的 C# 泛型 "any generic type"?

c# - 通用参数基类型 : "There is no implicit reference conversion from B to A"