C# 泛型類型參數的約束


在定義泛型類時,可以對客戶端代碼能夠在實例化類時用於類型參數的類型種類施加限制。如果客戶端代碼嘗試使用某個約束所不允許的類型來實例化類,則會產生編譯時錯誤。這些限制稱為約束。約束是使用 where 上下文關鍵字指定的。下表列出了六種類型的約束:

where T: struct
類型參數必須是值類型。可以指定除 Nullable 以外的任何值類型。有關更多信息,請參見使用可以為 null 的類型(C# 編程指南)。
where T : class
類型參數必須是引用類型;這一點也適用於任何類、接口、委托或數組類型。
where T:new()
類型參數必須具有無參數的公共構造函數。當與其他約束一起使用時,new() 約束必須最后指定。
where T:<基類名>
類型參數必須是指定的基類或派生自指定的基類。
where T:<接口名稱>
類型參數必須是指定的接口或實現指定的接口。可以指定多個接口約束。約束接口也可以是泛型的。
where T:U
為 T 提供的類型參數必須是為 U 提供的參數或派生自為 U 提供的參數。

如果要檢查泛型列表中的某個項以確定它是否有效,或者將它與其他某個項進行比較,則編譯器必須在一定程度上保證它需要調用的運算符或方法將受到客戶端代碼可能指定的任何類型參數的支持。這種保證是通過對泛型類定義應用一個或多個約束獲得的。例如,基類約束告訴編譯器:僅此類型的對象或從此類型派生的對象才可用作類型參數。一旦編譯器有了這個保證,它就能夠允許在泛型類中調用該類型的方法。約束是使用上下文關鍵字 where 應用的。

public class Employee
{
    private string name;
    private int id;

    public Employee(string s, int i)
    {
        name = s;
        id = i;
    }

    public string Name
    {
        get { return name; }
        set { name = value; }
    }

    public int ID
    {
        get { return id; }
        set { id = value; }
    }
}

public class GenericList<T> where T : Employee
{
    private class Node
    {
        private Node next;
        private T data;

        public Node(T t)
        {
            next = null;
            data = t;
        }

        public Node Next
        {
            get { return next; }
            set { next = value; }
        }

        public T Data
        {
            get { return data; }
            set { data = value; }
        }
    }

    private Node head;

    public GenericList() //constructor
    {
        head = null;
    }

    public void AddHead(T t)
    {
        Node n = new Node(t);
        n.Next = head;
        head = n;
    }

    public IEnumerator<T> GetEnumerator()
    {
        Node current = head;

        while (current != null)
        {
            yield return current.Data;
            current = current.Next;
        }
    }

    public T FindFirstOccurrence(string s)
    {
        Node current = head;
        T t = null;

        while (current != null)
        {
            //The constraint enables access to the Name property.
            if (current.Data.Name == s)
            {
                t = current.Data;
                break;
            }
            else
            {
                current = current.Next;
            }
        }
        return t;
    }
}

約束使得泛型類能夠使用 Employee.Name 屬性,因為類型為 T 的所有項都保證是 Employee 對象或從 Employee 繼承的對象。

可以對同一類型參數應用多個約束,並且約束自身可以是泛型類型,如下所示:

class EmployeeList<T> where T : Employee, IEmployee, System.IComparable<T>, new()
{
    // ...
}

 

通過約束類型參數,可以增加約束類型及其繼承層次結構中的所有類型所支持的允許操作和方法調用的數量。因此,在設計泛型類或方法時,如果要對泛型成員執行除簡單賦值之外的任何操作或調用 System.Object 不支持的任何方法,您將需要對該類型參數應用約束。

在應用 where T : class 約束時,避免對類型參數使用 == 和 != 運算符,因為這些運算符僅測試引用同一性而不測試值相等性。即使在用作參數的類型中重載這些運算符也是如此。下面的代碼說明了這一點;即使 String 類重載 == 運算符,輸出也為 false。

public static void OpTest<T>(T s, T t) where T : class
{
    System.Console.WriteLine(s == t);
}
static void Main()
{
    string s1 = "target";
    System.Text.StringBuilder sb = new System.Text.StringBuilder("target");
    string s2 = sb.ToString();
    OpTest<string>(s1, s2);
}

這種情況的原因在於,編譯器在編譯時僅知道 T 是引用類型,因此必須使用對所有引用類型都有效的默認運算符。如果必須測試值相等性,建議的方法是同時應用 where T : IComparable<T> 約束,並在將用於構造泛型類的任何類中實現該接口。

約束多個參數

可以對多個參數應用約束,並對一個參數應用多個約束,如下面的示例所示:

class Base { }
class Test<T, U>
    where U : struct
    where T : Base, new() { }

未綁定的類型參數

沒有約束的類型參數(如公共類 SampleClass<T>{} 中的 T)稱為未綁定的類型參數。未綁定的類型參數具有以下規則:

不能使用 != 和 == 運算符,因為無法保證具體類型參數能支持這些運算符。

可以在它們與 System.Object 之間來回轉換,或將它們顯式轉換為任何接口類型。

可以將它們與 null 進行比較。將未綁定的參數與 null 進行比較時,如果類型參數為值類型,則該比較將始終返回 false。

作為約束的類型參數

將泛型類型參數作為約束使用,在具有自己類型參數的成員函數必須將該參數約束為包含類型的類型參數時非常有用,如下示例所示:

class List<T>
{
    void Add<U>(List<U> items) where U : T {/*...*/}
}

在上面的示例中,T 在 Add 方法的上下文中是一個類型約束,而在 List 類的上下文中是一個未綁定的類型參數。

類型參數還可在泛型類定義中用作約束。請注意,必須在尖括號中聲明此類型參數與任何其他類型的參數:

//Type parameter V is used as a type constraint.
public class SampleClass<T, U, V> where T : V { }

泛型類的類型參數約束的作用非常有限,因為編譯器除了假設類型參數派生自 System.Object 以外,不會做其他任何假設。在希望強制兩個類型參數之間的繼承關系的情況下,可對泛型類使用參數類型約束。


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM