3

我有一些这样的课程

namespace ConsoleApplication1
{
    class Program
    {
        static void Main(string[] args)
        {
            A a = new C();

            a.method();
            Console.ReadLine();
        }
    }

    public class A
    {
        public virtual void method()
        {
            Console.WriteLine("METHOD FROM A");
        }
    }

    public class B : A { }

    public class C : B
    {
        public override void method()
        {
            Console.WriteLine("METHOD FROM C");
        }
    }
}

它工作正常,打印“METHOD FROM C”

如果我有这样的情况

namespace ConsoleApplication1
{
    class Program
    {
        static void Main(string[] args)
        {
            A a = new C();

            a.method();
            Console.ReadLine();
        }
    }

    public class A
    {
        public void method()
        {
            Console.WriteLine("METHOD FROM A");
        }
    }

    public class B : A { }

    public class C : B
    {
        public new void method()
        {
            Console.WriteLine("METHOD FROM C");
        }
    }
}

它打印“来自 A 的方法”。如何在不采用强制转换或更改方法声明的情况下获得与第一个示例相同的行为?

4

3 回答 3

11

你不能 - 这种行为差异是使用虚拟/覆盖的重点。

当你用“new”声明一个方法时,你是在说“我知道我隐藏了一个具有相同签名的方法,而不是覆盖它;我不想要多态行为。”

同样,当您声明一个方法而不指定“虚拟”时,您是在说“我不希望子类能够覆盖此方法”。

你为什么要这样做?您实际上是否只是试图覆盖尚未声明为虚拟的方法?如果是这样,就没有办法绕过它——而且有充分的理由。如果作者在设计类时没有考虑到多态性,那么如果您能够覆盖该方法,它很容易被破坏。

当然,如果你将变量声明为子类的类型,像这样:

C c = new C();
c.method();

那么这调用新声明的方法。

于 2010-03-02T10:20:02.890 回答
2

就像乔恩说的那样避免它。但

static void Main(string[] args)
    {
        A a = new C();

        (a as C).method();

        Console.ReadLine();
    }
于 2010-03-02T10:28:58.800 回答
1

您可以使用动态关键字调用此方法。

namespace ConsoleApplication1
{
    class Program
    {
        static void Main(string[] args)
        {
            dynamic a = new C();

            a.method();
            Console.ReadLine();
        }
    }

    public class A
    {
        public void method()
        {
            Console.WriteLine("METHOD FROM A");
        }
    }

    public class B : A { }

    public class C : B
    {
        public new void method()
        {
            Console.WriteLine("METHOD FROM C");
        }
    }
}
于 2012-12-10T09:01:02.607 回答