Partial 4 страница. class Shape: ICloneable { object ICloneable.Clone() {} }
class Shape: ICloneable class Ellipse: Shape объявление метода ICloneable.Clone в классе Ellipse приведет к ошибке при компилировании, поскольку интерфейс ICloneable не указан явно в списке базовых классов класса Ellipse. Полное имя интерфейса должно указывать интерфейс, в котором был объявлен этот член. Так, в объявлениях interface IControl interface ITextBox: IControl class TextBox: ITextBox void ITextBox.SetText(string text) {...} явная реализация члена интерфейса Paint должна быть записана в форме IControl.Paint. 13.4.2 Уникальность реализованных интерфейсов Интерфейсы, реализованные в объявлении уникального типа, должны оставаться уникальными для всех сформированных типов. Если бы это правило не действовало, было бы невозможно определить правильный метод для вызова в конкретных сформированных типах. Например, предположим, что объявление универсального класса может иметь следующий вид: interface I<T> class X<U,V>: I<U>, I<V> //Ошибка: интерфейсы I<U> и I<V> конфликтуют друг с другом В результате было бы невозможно определить, какой код должен выполняться в следующем случае: I<int> x = new X<int,int>(); Чтобы определить, является ли список объявлений универсального типа допустимым, требуется выполнить следующие шаги: · L — список интерфейсов, непосредственно определенных в универсальном объявлении класса, структуры или интерфейса C. · Добавить в L все базовые интерфейсы для интерфейсов, уже содержащихся в L. · Удалить из L все дубликаты · Если после создания на основе C любого возможного типа в результате подстановки в L аргументов типа два интерфейса в L станут идентичными, такое объявление C является недопустимым. При определении всех возможных сформированных типов не учитываются объявления ограничений. В приведенном выше объявлении класса X список интерфейсов L состоит из интерфейсов I<U> и I<V>. Объявление является недопустимым, поскольку в любом сформированном типе, где U и V имеют одинаковый тип, эти два интерфейса также будут иметь одинаковый тип. Можно объединять интерфейсы, находящиеся на разных уровнях наследования: interface I<T> class Base<U>: I<U> class Derived<U,V>: Base<U>, I<V> // ОК Этот код является допустимым, несмотря на то, что в классе Derived<U,V> реализуются как интерфейс I<U>, так и интерфейс I<V>. Код I<int> x = new Derived<int,int>(); вызовет метод класса Derived, поскольку класс Derived<int,int> фактически выполняет I<int> (§13.4.6). 13.4.3 Реализация универсальных методов Если в универсальном методе неявно реализуется метод интерфейса, то ограничения, заданные для каждого из параметров типа метода, должны быть одинаковы в обоих объявлениях (после замены любых параметров типа интерфейса соответствующими аргументами типа), где параметры типа метода указываются их положением слева направо. Тем не менее, если в универсальном методе явно реализуется метод интерфейса, ограничения для реализующего метода не допускаются. Эти ограничения наследуются из метода интерфейса interface I<A,B,C> class C: I<object,C,string> Метод C.F<T> неявно реализует метод I<object,C,string>.F<T>. В этом случае в методе C.F<T> не требуется (и не допускается) указывать ограничение T: object, так как object является неявным ограничением для всех параметров типа. Метод C.G<T> неявно реализует метод I<object,C,string>.G<T>, поскольку ограничения совпадают с установленными в интерфейсе после замены параметров типа интерфейса соответствующими аргументами типа. Ограничение для метода C.H<T> является ошибочным, так как запечатанные типы (в данном случае string) не могут использоваться как ограничения. Отсутствие ограничения также было бы ошибкой, поскольку ограничения в неявных реализациях метода интерфейса должны совпадать. Таким образом, выполнить неявную реализацию метода I<object,C,string>.H<T> невозможно. Этот метод интерфейса может быть реализован только с использованием явной реализации члена интерфейса: class C: I<object,C,string> public void H<U>(U u) where U: class {...} void I<object,C,string>.H<T>(T t) { В этом примере явная реализация члена интерфейса вызывает открытый метод с применением более мягких ограничений. Обратите внимание, что присваивание из t в s является допустимым, поскольку T наследует ограничение T: string, хотя это ограничение и не указано в исходном коде. 13.4.4 Сопоставление интерфейсов В классе или структуре должны содержаться реализации всех членов интерфейсов, включенных в список базовых классов такого класса или структуры. Процедура выявления реализаций членов интерфейса в реализующих их классе или структуре называется сопоставлением интерфейсов. При сопоставлении интерфейсов для класса или структуры C выявляется реализация каждого из интерфейсов, указанных в списке базовых классов C. Реализация конкретного члена интерфейса I.M, где I указывает интерфейс, в котором объявлен член M, определяется путем проверки каждого класса или структуры S, начиная с C и последовательно проходя по каждому базовому классу C до выявления совпадения: · Если S содержит объявление явной реализации члена интерфейса, которая соответствует I и M, этот член является реализацией члена I.M. · В обратном случае, если S содержит объявление открытого члена, не являющегося статическим, который соответствует M, то реализацией члена I.M является этот член. При выявлении совпадения для нескольких членов не указывается, какой из них является реализацией члена I.M. Эта ситуация может произойти только в том случае, если S относится к сформированному типу, в котором два члена, согласно объявлению в универсальном типе, имеют разные подписи, однако аргументы типа делают их подписи идентичными.
|