Anteckning
Åtkomst till den här sidan kräver auktorisering. Du kan prova att logga in eller ändra kataloger.
Åtkomst till den här sidan kräver auktorisering. Du kan prova att ändra kataloger.
Med ett covariant-gränssnitt kan dess metoder returnera fler härledda typer än de som anges i gränssnittet. Med ett kontravariant gränssnitt kan dess metoder acceptera parametrar av mindre härledda typer än de som anges i gränssnittet.
I .NET Framework 4 blev flera befintliga gränssnitt samtidiga och kontravarianta. Dessa inkluderar IEnumerable<T> och IComparable<T>. På så sätt kan du återanvända metoder som fungerar med generiska samlingar av bastyper för samlingar av härledda typer.
En lista över variantgränssnitt i .NET finns i Varians i Generiska gränssnitt (C#).
Konvertera allmänna samlingar
I följande exempel visas fördelarna med samvariansstöd i IEnumerable<T> gränssnittet. Metoden PrintFullName accepterar en samling av typen IEnumerable<Person> som en parameter. Du kan dock återanvända den för en samling av typen IEnumerable<Employee> eftersom Employee ärver Person.
// Simple hierarchy of classes.
public class Person
{
public string FirstName { get; set; }
public string LastName { get; set; }
}
public class Employee : Person { }
class Program
{
// The method has a parameter of the IEnumerable<Person> type.
public static void PrintFullName(IEnumerable<Person> persons)
{
foreach (Person person in persons)
{
Console.WriteLine("Name: {0} {1}",
person.FirstName, person.LastName);
}
}
public static void Test()
{
IEnumerable<Employee> employees = new List<Employee>();
// You can pass IEnumerable<Employee>,
// although the method expects IEnumerable<Person>.
PrintFullName(employees);
}
}
Jämföra generiska samlingar
I följande exempel visas fördelarna med kontravariansstöd i IEqualityComparer<T> gränssnittet. Klassen PersonComparer implementerar gränssnittet IEqualityComparer<Person>. Du kan dock återanvända den här klassen för att jämföra en sekvens med objekt av typen Employee eftersom Employee ärver Person.
// Simple hierarchy of classes.
public class Person
{
public string FirstName { get; set; }
public string LastName { get; set; }
}
public class Employee : Person { }
// The custom comparer for the Person type
// with standard implementations of Equals()
// and GetHashCode() methods.
class PersonComparer : IEqualityComparer<Person>
{
public bool Equals(Person x, Person y)
{
if (Object.ReferenceEquals(x, y)) return true;
if (Object.ReferenceEquals(x, null) ||
Object.ReferenceEquals(y, null))
return false;
return x.FirstName == y.FirstName && x.LastName == y.LastName;
}
public int GetHashCode(Person person)
{
if (Object.ReferenceEquals(person, null)) return 0;
int hashFirstName = person.FirstName == null
? 0 : person.FirstName.GetHashCode();
int hashLastName = person.LastName.GetHashCode();
return hashFirstName ^ hashLastName;
}
}
class Program
{
public static void Test()
{
List<Employee> employees = new List<Employee> {
new Employee() {FirstName = "Michael", LastName = "Alexander"},
new Employee() {FirstName = "Jeff", LastName = "Price"}
};
// You can pass PersonComparer,
// which implements IEqualityComparer<Person>,
// although the method expects IEqualityComparer<Employee>.
IEnumerable<Employee> noduplicates =
employees.Distinct<Employee>(new PersonComparer());
foreach (var employee in noduplicates)
Console.WriteLine(employee.FirstName + " " + employee.LastName);
}
}