C# Language Tipo de restricciones (clases e interfaces)


Ejemplo

Las restricciones de tipo pueden forzar un parámetro de tipo para implementar una determinada interfaz o clase.

interface IType;
interface IAnotherType;

// T must be a subtype of IType
interface IGeneric<T>
    where T : IType
{
}

// T must be a subtype of IType
class Generic<T>
    where T : IType
{
}

class NonGeneric
{
    // T must be a subtype of IType
    public void DoSomething<T>(T arg)
        where T : IType
    {
    }
}

// Valid definitions and expressions:
class Type : IType { }
class Sub : IGeneric<Type> { }
class Sub : Generic<Type> { }
new NonGeneric().DoSomething(new Type());

// Invalid definitions and expressions:
class AnotherType : IAnotherType { }
class Sub : IGeneric<AnotherType> { }
class Sub : Generic<AnotherType> { }
new NonGeneric().DoSomething(new AnotherType());

Sintaxis para múltiples restricciones:

class Generic<T, T1>
    where T : IType 
    where T1 : Base, new()
{
}

Las restricciones de tipo funcionan de la misma manera que la herencia, ya que es posible especificar múltiples interfaces como restricciones en el tipo genérico, pero solo una clase:

class A { /* ... */ }
class B { /* ... */ }

interface I1 { }
interface I2 { }

class Generic<T>
    where T : A, I1, I2
{
}

class Generic2<T>
    where T : A, B //Compilation error
{
}

Otra regla es que la clase debe agregarse como la primera restricción y luego las interfaces:

class Generic<T>
    where T : A, I1
{
}

class Generic2<T>
    where T : I1, A //Compilation error
{
}

Todas las restricciones declaradas deben satisfacerse simultáneamente para que funcione una instanciación genérica particular. No hay manera de especificar dos o más conjuntos alternativos de restricciones.