Megjegyzés
Az oldalhoz való hozzáféréshez engedély szükséges. Megpróbálhat bejelentkezni vagy módosítani a címtárat.
Az oldalhoz való hozzáféréshez engedély szükséges. Megpróbálhatja módosítani a címtárat.
A kovarians felület lehetővé teszi a metódusok számára, hogy az interfészben megadottaknál több származtatott típust adjanak vissza. A contravariant interfész lehetővé teszi a metódusok számára, hogy az interfészben megadottaknál kevésbé származtatott típusú paramétereket fogadjanak el.
A .NET-keretrendszer 4-ben több meglévő interfész kovariant és contravariant lett. Ezek közé tartoznak a következők: IEnumerable<T> és IComparable<T>. Ez lehetővé teszi a származtatott típusok gyűjteményeinek általános alaptípus-gyűjteményeivel működő metódusok újrafelhasználását.
A .NET-beli változatillesztők listáját lásd: Variancia az általános felületekben (C#).
Általános gyűjtemények konvertálása
Az alábbi példa a kovariancia-támogatás előnyeit mutatja be a IEnumerable<T> felületen. A PrintFullName metódus paraméterként fogadja el a IEnumerable<Person> típus gyűjteményét. Azonban újrafelhasználhatja a IEnumerable<Employee> típusú gyűjteményhez, mert a Employee örökli a 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);
}
}
Általános gyűjtemények összehasonlítása
Az alábbi példa a contravariance támogatás előnyeit mutatja be a IEqualityComparer<T> felületen. Az PersonComparer osztály megvalósítja az IEqualityComparer<Person> felületet. Ezt az osztályt azonban újra felhasználhatja az ilyen típusú objektumok sorozatának összehasonlításához, mert Employee örökli azokat EmployeePerson.
// 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);
}
}