From: 011netservice@gmail.com Date: 2022-04-24 Subject: Where-GenericTypeConstraint.txt 泛型型別條件約束關鍵字 where (generic type constraint) ---------- 20200805 Example: 限制 (Genernic Type) 為 指定的型別: public class UsingEnum where T : System.Enum { } public class UsingDelegate where T : System.Delegate { } public class Multicaster where T : System.MulticastDelegate { } 限制 (多個 Genernic Type) 為 指定的型別: class MyClass where T : class where U : struct { } 限制 (Genernic Type) 為 non-nullable types: 可搭配 nullable define 開啟編譯器警示 The notnull constraint is available starting in C# 8.0 for code compiled in a nullable enable context. Unlike other constraints, if a type argument violates the notnull constraint, the compiler generates a warning instead of an error. Warnings are only generated in a nullable enable context. #nullable enable class NotNullContainer where T : notnull { } #nullable restore 限制 (Genernic Type) 為 unmanaged types class UnManagedWrapper where T : unmanaged { } Class 加入 where T : new() 後, 才能以 new T() 語法, 新建指定型別的物件. 開啟 New() 功能, 才能新建(Genernic Type)物件: public class MyGenericClass where T : IComparable, new() { // The following line is not possible without new() constraint: T item = new T(); } 限制 (多個 Genernic Type) 為 指定的型別: public interface IMyInterface { } namespace CodeExample { class Dictionary where TKey : IComparable where TVal : IMyInterface { public void Add(TKey key, TVal val) { } } } 限制 (Genernic Type) 為 包含指定的方法: attach constraints to type parameters of generic methods public void MyMethod(T t) where T : IMyInterface { } 使用在 delegate 型別的語法, 跟 (限制指定方法) 相同: Notice that the syntax to describe type parameter constraints on delegates is the same as that of methods: delegate T MyDelegate() where T : new(); ---------- 20200805 where (generic type constraint) The where clause in a generic definition specifies constraints on the types that are used as arguments for type parameters in a generic type, method, delegate, or local function. Constraints can specify interfaces, base classes, or require a generic type to be a reference, value, or unmanaged type. They declare capabilities that the type argument must have. For example, you can declare a generic class, MyGenericClass, such that the type parameter T implements the IComparable interface: C# 複製 public class AGenericClass where T : IComparable { } 注意 For more information on the where clause in a query expression, see where clause. The where clause can also include a base class constraint. The base class constraint states that a type to be used as a type argument for that generic type has the specified class as a base class, or is that base class. If the base class constraint is used, it must appear before any other constraints on that type parameter. Some types are disallowed as a base class constraint: Object, Array, and ValueType. Before C# 7.3, Enum, Delegate, and MulticastDelegate were also disallowed as base class constraints. The following example shows the types that can now be specified as a base class: C# 複製 public class UsingEnum where T : System.Enum { } public class UsingDelegate where T : System.Delegate { } public class Multicaster where T : System.MulticastDelegate { } In a nullable context in C# 8.0 and later, the nullability of the base class type is enforced. If the base class is non-nullable (for example Base), the type argument must be non-nullable. If the base class is nullable (for example Base?), the type argument may be either a nullable or non-nullable reference type. The compiler issues a warning if the type argument is a nullable reference type when the base class is non-nullable. The where clause can specify that the type is a class or a struct. The struct constraint removes the need to specify a base class constraint of System.ValueType. The System.ValueType type may not be used as a base class constraint. The following example shows both the class and struct constraints: C# 複製 class MyClass where T : class where U : struct { } In a nullable context in C# 8.0 and later, the class constraint requires a type to be a non-nullable reference type. To allow nullable reference types, use the class? constraint, which allows both nullable and non-nullable reference types. The where clause may include the notnull constraint. The notnull constraint limits the type parameter to non-nullable types. That type may be a value type or a non-nullable reference type. The notnull constraint is available starting in C# 8.0 for code compiled in a nullable enable context. Unlike other constraints, if a type argument violates the notnull constraint, the compiler generates a warning instead of an error. Warnings are only generated in a nullable enable context. 重要 Generic declarations that include the notnull constraint can be used in a nullable oblivious context, but compiler does not enforce the constraint. C# 複製 #nullable enable class NotNullContainer where T : notnull { } #nullable restore The where clause may also include an unmanaged constraint. The unmanaged constraint limits the type parameter to types known as unmanaged types. The unmanaged constraint makes it easier to write low-level interop code in C#. This constraint enables reusable routines across all unmanaged types. The unmanaged constraint can't be combined with the class or struct constraint. The unmanaged constraint enforces that the type must be a struct: C# 複製 class UnManagedWrapper where T : unmanaged { } The where clause may also include a constructor constraint, new(). That constraint makes it possible to create an instance of a type parameter using the new operator. The new() Constraint lets the compiler know that any type argument supplied must have an accessible parameterless constructor. For example: C# 複製 public class MyGenericClass where T : IComparable, new() { // The following line is not possible without new() constraint: T item = new T(); } The new() constraint appears last in the where clause. The new() constraint can't be combined with the struct or unmanaged constraints. All types satisfying those constraints must have an accessible parameterless constructor, making the new() constraint redundant. With multiple type parameters, use one where clause for each type parameter, for example: C# 複製 public interface IMyInterface { } namespace CodeExample { class Dictionary where TKey : IComparable where TVal : IMyInterface { public void Add(TKey key, TVal val) { } } } You can also attach constraints to type parameters of generic methods, as shown in the following example: C# 複製 public void MyMethod(T t) where T : IMyInterface { } Notice that the syntax to describe type parameter constraints on delegates is the same as that of methods: C# 複製 delegate T MyDelegate() where T : new(); For information on generic delegates, see Generic Delegates. For details on the syntax and use of constraints, see Constraints on Type Parameters.