Явная реализация интерфейса (Руководство по программированию в C#)Explicit Interface Implementation (C# Programming Guide)

Если класс реализует два интерфейса, содержащих член с одинаковой сигнатурой, при реализации такого члена в классе оба интерфейса будут использовать этот член в качестве собственной реализации.If a class implements two interfaces that contain a member with the same signature, then implementing that member on the class will cause both interfaces to use that member as their implementation. В следующем примере все вызовы Paint приводят к вызову одного и того же метода.In the following example, all the calls to Paint invoke the same method. Первый пример определяет типы:This first sample defines the types:

public interface IControl
{
    void Paint();
}
public interface ISurface
{
    void Paint();
}
public class SampleClass : IControl, ISurface
{
    // Both ISurface.Paint and IControl.Paint call this method.
    public void Paint()
    {
        Console.WriteLine("Paint method in SampleClass");
    }
}

Следующий пример вызывает методы:The following sample calls the methods:

SampleClass sample = new SampleClass();
IControl control = sample;
ISurface surface = sample;

// The following lines all call the same method.
sample.Paint();
control.Paint();
surface.Paint();
// Output:
// Paint method in SampleClass
// Paint method in SampleClass
// Paint method in SampleClass

Если два члена интерфейса выполняют разные функции, это может привести к некорректной реализации одного или обоих интерфейсов.When two interface members don't perform the same function, it leads to an incorrect implementation of one or both of the interfaces. Член интерфейса можно реализовать явно, создав член класса, который вызывается только через этот интерфейс и относится только к нему.It's possible to implement an interface member explicitly—creating a class member that is only called through the interface, and is specific to that interface. Укажите в имени члена класса имя интерфейса и точку.Name the class member with the name of the interface and a period. Пример:For example:

public class SampleClass : IControl, ISurface
{
    void IControl.Paint()
    {
        System.Console.WriteLine("IControl.Paint");
    }
    void ISurface.Paint()
    {
        System.Console.WriteLine("ISurface.Paint");
    }
}

Член класса IControl.Paint доступен только через интерфейс IControl, а ISurface.Paint — только через интерфейс ISurface.The class member IControl.Paint is only available through the IControl interface, and ISurface.Paint is only available through ISurface. Обе реализации метода будут разделены, и ни одна из них не будет доступна в классе напрямую.Both method implementations are separate, and neither are available directly on the class. Пример:For example:

// Call the Paint methods from Main.

SampleClass obj = new SampleClass();
//obj.Paint();  // Compiler error.

IControl c = obj;
c.Paint();  // Calls IControl.Paint on SampleClass.

ISurface s = obj;
s.Paint(); // Calls ISurface.Paint on SampleClass.

// Output:
// IControl.Paint
// ISurface.Paint

Явная реализация также применяется в случаях, когда в каждом из двух интерфейсов объявляются разные члены (например, свойство и метод) с одинаковыми именами.Explicit implementation is also used to resolve cases where two interfaces each declare different members of the same name such as a property and a method. Чтобы реализовать оба интерфейса и избежать ошибок компилятора, класс должен использовать явную реализацию либо свойства P, либо метода P, либо одновременно обоих этих членов.To implement both interfaces, a class has to use explicit implementation either for the property P, or the method P, or both, to avoid a compiler error. Пример:For example:

interface ILeft
{
    int P { get;}
}
interface IRight
{
    int P();
}

class Middle : ILeft, IRight
{
    public int P() { return 0; }
    int ILeft.P { get { return 0; } }
}

Начиная с версии C# 8.0, вы можете определять реализацию для членов, объявленных в интерфейсе.Beginning with C# 8.0, you can define an implementation for members declared in an interface. Если класс наследует реализацию метода от интерфейса, этот метод доступен только через ссылку типа интерфейса.If a class inherits a method implementation from an interface, that method is only accessible through a reference of the interface type. Наследуемый член не отображается как часть открытого интерфейса.The inherited member doesn't appear as part of the public interface. Следующий пример определяет реализацию по умолчанию для метода интерфейса:The following sample defines a default implementation for an interface method:

public interface IControl
{
    void Paint() => Console.WriteLine("Default Paint method");
}
public class SampleClass : IControl
{
    // Paint() is inherited from IControl.
}

Следующий пример вызывает реализацию по умолчанию:The following sample invokes the default implementation:

var sample = new SampleClass();
//sample.Paint();// "Paint" isn't accessible.
var control = sample as IControl;
control.Paint();

Любой класс, реализующий интерфейс IControl, может переопределить метод Paint по умолчанию либо в качестве открытого метода, либо в качестве реализации интерфейса.Any class that implements the IControl interface can override the default Paint method, either as a public method, or as an explicit interface implementation.

См. такжеSee also