Разница между затенением и переопределением в С#?

Какая разница между затенением и переопределением метода в С#?

Ответы

Ответ 1

Наследование наследования...

Предположим, что у вас есть эти классы:

class A {
   public int Foo(){ return 5;}
   public virtual int Bar(){return 5;}
}
class B : A{
   public new int Foo() { return 1;}     //shadow
   public override int Bar() {return 1;} //override
}

тогда, когда вы вызываете это:

A clA = new A();
B clB = new B();

Console.WriteLine(clA.Foo()); // output 5
Console.WriteLine(clA.Bar()); // output 5
Console.WriteLine(clB.Foo()); // output 1
Console.WriteLine(clB.Bar()); // output 1

//now let cast B to an A class
Console.WriteLine(((A)clB).Foo()); // output 5 <<<-- shadow
Console.WriteLine(((A)clB).Bar()); // output 1

Предположим, что у вас есть базовый класс, и вы используете базовый класс во всем своем коде вместо унаследованных классов, и используете тень, он вернет значения, возвращаемые базовым классом, вместо того, чтобы следовать дереву наследования реального типа объект.

Выполнить код здесь

Надеюсь, у меня есть смысл:)

Ответ 2

Затенение на самом деле является языком VB для того, что мы будем называть скрытым в С#.

Часто скрытие (затенение в VB) и переопределение отображаются в ответе Stormenet.

Показано, что виртуальный метод переопределяется подклассом, и вызовы этого метода даже по типу суперкласса или из внутреннего кода суперкласса вызовут реализацию замены из подкласса.

Затем отображается конкретный метод (который не помечен как виртуальный или абстрактный) скрывается с помощью ключевого слова new при определении метода с идентичной сигнатурой в подклассе. В этом случае, когда метод вызывается в классе суперкласса, используется исходная реализация, новая реализация доступна только в подклассе.

Однако часто пропускается то, что также можно скрыть виртуальный метод.

class A
{
    public virtual void DoStuff() { // original implementation }
}

class B : A
{
    public new void DoStuff() {  //new implementation }
}

B b = new B();
A a = b;

b.DoStuff(); //calls new implementation
a.DoStuff(); //calls original implementation.

Примечание в приведенном выше примере DoStuff становится конкретным и не может быть переопределен. Однако также возможно использовать как ключевые слова virtual, так и new.

class A
{
    public virtual void DoStuff() { // original implementation }
}

class B : A
{
    public new virtual void DoStuff() {  //new implementation }
}

class C : B
{
    public override void DoStuff() { //replacement implementation }
}

C c = new C();
B b = c;
A a = b;

c.DoStuff(); //calls replacement implementation
b.DoStuff(); //calls replacement implementation
a.DoStuff(); //calls original implementation.

Обратите внимание, что, несмотря на то, что все используемые методы являются виртуальными, переопределение на C не влияет на виртуальный метод на A, поскольку использование new в B скрывает реализацию A.

Изменить: В комментариях к этому ответу было отмечено, что вышеупомянутое может быть опасным или, по крайней мере, не особенно полезным. Я бы сказал, да, это может быть опасно и будет там, если бы это было вообще полезно.

В частности, вы можете столкнуться со всеми неприятностями, если вы также измените модификаторы доступности. Например: -

public class Foo
{
    internal Foo() { }
    protected virtual string Thing() { return "foo"; }
}

public class Bar : Foo
{
 internal new string Thing() { return "bar"; }
}

Для внешнего наследника Bar, Foo реализация Thing() остается доступной и переопределяемой. Все юридические и объяснимые в соответствии с правилами типа .NET никогда не бывают совершенно неинтуитивными.

Я опубликовал этот ответ, чтобы углубить понимание того, как все работает не как предложение методов, которые могут быть использованы свободно.

Ответ 3

Я думаю, что основное отличие заключается в том, что при затенении вы фактически используете имя и просто игнорируете использование суперкласса. При переопределении вы меняете реализацию, но не доступность и подпись (например, типы параметров и возврат). См. http://www.geekinterview.com/question_details/19331.

Ответ 4

Тень - концепция VB.NET. В С# Shadowing известен как Hiding. Он скрывает метод производного класса. Это выполняется с использованием нового ключевого слова.

Ключевое слово Override используется для обеспечения полностью новой реализации метода базового класса (который помечен как "Виртуальный" ) в производном классе.

Ответ 5

В принципе, если у вас есть что-то вроде ниже,

Class A
{
}

Class B:A
{
}

A a = new B();

Любой метод, который вы вызываете для объекта "a", будет выполняться по типу "a" (здесь тип "A" ) Но если вы реализуете тот же метод в классе B, который уже присутствует в классе A, компилятор даст вам предупреждение использовать ключевое слово "Новое". Если вы используете "Новый", предупреждение исчезнет. Помимо этого нет никакой разницы между использованием "New" или не использованием его в унаследованном классе.

В некоторых ситуациях вам может потребоваться вызвать метод ссылочного класса, который конкретный экземпляр имеет место в тот момент, вместо вызова метода по типу объекта. В приведенном выше случае эта ссылка имеет значение "B", но тип "A". Поэтому, если вы хотите, чтобы вызов метода выполнялся на "B", вы используете Virtual и переопределяете его для достижения.

Надеюсь, что это поможет...

Даниэль Сандип.