Detaillierte Erklärung von Kontravarianz und Kovarianz in C#

黄舟
Freigeben: 2017-09-02 14:32:04
Original
1243 Leute haben es durchsucht

Dieser Artikel stellt hauptsächlich die relevanten Informationen zur C#-Inversion und -Kovarianz im Detail vor, die einen gewissen Referenzwert haben.

Dieser Artikel verwendet weitere Delegierte und Lambda-Ausdrücke Schauen Sie sich hierzu bitte meine Artikel „Delegierte und anonyme Delegierte“ und „Anonyme Delegierte und Lambda-Ausdrücke“ an, um Ihnen beim Aufbau eines vollständigen Wissenssystems zu helfen.

Im Prozess von C# von seiner Geburt bis zu seiner Entwicklung und seinem Wachstum werden ständig neue Wissenspunkte eingeführt. Kontravarianz und Kovarianz sind keine Originalität von C# und werden später eingeführt. Kontravarianz und Kovarianz gibt es auch in Java. Ich werde in Zukunft auch einen Artikel über Java-Kontravarianz und Kovarianz schreiben.

Kontravarianz und Kovarianz klingen abstrakt und tiefgreifend, sind aber eigentlich sehr einfach. Schauen Sie sich den folgenden Code an:


class Person
 {

 }
 class Student : Person
 {

 }
 class Teacher: Person
 {

 }
 
 class Program
 {
  static void Main(string[] args)
  {
   List<Person> plist = new List<Person>();
   plist = new List<Student>();
   plist = new List<Teacher>();
}
}
Nach dem Login kopieren

Im obigen Code ist plist = new List(), plist = new List() Eine Zusammenstellung Fehler tritt auf. Obwohl „Person“ die übergeordnete Klasse von „Student/Teacher“ ist, ist der Typ „List“ nicht die übergeordnete Klasse des Typs „List“. Daher meldet die obige Zuweisungsanweisung einen Typkonvertierungsfehler.

Zuweisungsvorgänge wie die oben genannten waren vor C# 4.0 nicht zulässig. Der Grund dafür, warum sie nicht zulässig sind, ist die Typensicherheit. Schauen Sie sich den folgenden Beispielcode an:


List<Person> plist = new List<Student>();
plist.Add(new Person());
plist.Add(new Student());
plist.Add(new Teacher());
Nach dem Login kopieren

Das folgende Beispiel geht davon aus, dass List plist = new List() eine Zuweisung zulässt, obwohl der Typ von plist ist die List< ;Person>-Sammlung, verweist jedoch tatsächlich auf die List-Sammlung. plist.Add(new Person()), die eigentliche Additionsoperation ruft List.Add() auf. Der Typ „Person“ kann nicht sicher in „Student“ konvertiert werden, daher ist eine solche Sammlungsdefinition nicht sinnvoll und die obige Annahme trifft daher nicht zu.

Aber die Situation hat sich nach C# 4.0 geändert. Es ist nicht so, dass „unmögliche Dinge passiert sind“, sondern dass die Flexibilität der Anwendung neue Anpassungen vorgenommen hat. Ebenso ist das obige Programm in C# 4.0 immer noch nicht zulässig, es tritt jedoch eine Ausnahme auf. Ab C# 4.0 dürfen in generischen Delegaten und generischen Schnittstellen Sondersituationen auftreten (im Wesentlichen sind keine besonderen Änderungen aufgetreten, die später erläutert werden). Das folgende Beispiel:


delegate void Work<T>(T item);

class Person
{
  public string Name { get; set; }
}
class Student : Person
{
  public string Like { get; set; }
}
class Teacher : Person
{
  public string Teach { get; set; }
}

class Program
{
  static void Main(string[] args)
  {
   Work<Person> worker = (p) => { Console.WriteLine(p.Name); }; ;
   Work<Student> student_worker = (s) => { Console.WriteLine(s.Like); };
   student_worker = worker; //此处编译错误
  }
}
Nach dem Login kopieren

Nach der bisherigen theoretischen Unterstützung ist der Fehler von student_worker = worker leicht zu verstehen. Der Zweck unseres Programms hier besteht jedoch darin, den Arbeiter als Work fungieren zu lassen. In Zukunft wird der Aufruf von student_worker(s) tatsächlich Waker(s) aufrufen. Um unsere Anforderungen zu erfüllen, muss das Programm zwei Aspekte der Verarbeitung ausführen:

1 Denn beim Aufruf von student_worker(s) werden tatsächlich Waker(s) ausgeführt, also der Typ der s-Variablen muss erfolgreich in den vom Worker benötigten Parametertyp konvertiert werden.

2. Sie müssen dem Compiler mitteilen, dass er Variablen vom Typ Work zuweisen darf.

Bedingung 1 Beim Aufruf von student_worker() fordert der Compiler an, dass der Parameter ein Objekt vom Typ „Student“ sein muss, das erfolgreich in ein Objekt vom Typ „Person“ konvertiert werden kann.

Bedingung 2 erfordert Anpassungen an der Woke-Delegiertendefinition. Die Anpassungen sind wie folgt:


delegate void WorkIn<in T>(T item);
Nach dem Login kopieren

Der Zweck der Änderung des Delegiertennamens in WorkIn ist Um Änderungen vor und nach der Delegation zu vermeiden, ist der entscheidende Punkt . Markieren Sie durch Hinzufügen des Schlüsselworts in den Typparameter T des generischen Delegaten und verwenden Sie ihn nur als Parameter der Delegatenmethode. An diesem Punkt kann das obige Programm erfolgreich kompiliert und ausgeführt werden.


delegate void WorkIn<in T>(T item);
class Program
 {
  static void Main(string[] args)
  {
   WorkIn woker = (p) => { Console.WriteLine(p.Name); };
   WorkIn student_worker = woker;
   student_worker(new Student() { Name="tom", Like="C#" });

  }
 }
Nach dem Login kopieren

Die Situation, in der Typparameter Subtypen sein müssen und Zuweisungstypparameter übergeordnete Typwerte sein können, wird als Kontravarianz bezeichnet. Kontravarianz erfordert, dass die Typparameter von Generika in C# markiert werden. Obwohl Kontravarianz als Kontravarianz bezeichnet wird, sieht es nur formal so aus, als ob das übergeordnete Klassenobjekt der Unterklassenvariablen zugewiesen wird. Im Wesentlichen handelt es sich um die Typkonvertierung der Parameter beim Aufruf der Methode. Student s = new Person(), das ist unmöglich, das ist keine Kontravariante, es ist ein Fehler.

Wenn Sie den obigen Code in die folgende Form umwandeln können, können Sie die Inversion vergessen. Das Wesentliche ist wichtiger als das Phänomen.

Das obige ist der detaillierte Inhalt vonDetaillierte Erklärung von Kontravarianz und Kovarianz in C#. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Verwandte Etiketten:
Quelle:php.cn
Erklärung dieser Website
Der Inhalt dieses Artikels wird freiwillig von Internetnutzern beigesteuert und das Urheberrecht liegt beim ursprünglichen Autor. Diese Website übernimmt keine entsprechende rechtliche Verantwortung. Wenn Sie Inhalte finden, bei denen der Verdacht eines Plagiats oder einer Rechtsverletzung besteht, wenden Sie sich bitte an admin@php.cn
Beliebte Tutorials
Mehr>
Neueste Downloads
Mehr>
Web-Effekte
Quellcode der Website
Website-Materialien
Frontend-Vorlage
Über uns Haftungsausschluss Sitemap
Chinesische PHP-Website:Online-PHP-Schulung für das Gemeinwohl,Helfen Sie PHP-Lernenden, sich schnell weiterzuentwickeln!