7

このコードがなぜそのように動作するのか誰かに教えてもらえますか?コードに埋め込まれたコメントを参照してください...

私はここで本当に明白な何かを逃していますか?

using System;
namespace ConsoleApplication3
{
    public class Program
    {
        static void Main(string[] args)
        {
            var c = new MyChild();
            c.X();
            Console.ReadLine();
        }
    }

    public class MyParent
    {
        public virtual void X()
        {
            Console.WriteLine("Executing MyParent");
        }
    }

    delegate void MyDelegate();

    public class MyChild : MyParent
    {
        public override void X()
        {
            Console.WriteLine("Executing MyChild");
            MyDelegate md = base.X;

            // The following two calls look like they should behave the same,
            //  but they behave differently!    

            // Why does Invoke() call the base class as expected here...
            md.Invoke();

            // ... and yet BeginInvoke() performs a recursive call within
            //  this child class and not call the base class?
            md.BeginInvoke(CallBack, null);
        }

        public void CallBack(IAsyncResult iAsyncResult)
        {
            return;
        }
    }
}
4

3 に答える 3

5

私はまだ答えを持っていませんが、奇妙さを示すための少し明確なプログラムであると私が信じているものがあります:

using System;

delegate void MyDelegate();

public class Program
{
    static void Main(string[] args)
    {
        var c = new MyChild();
        c.DisplayOddity();
        Console.ReadLine();
    }
}

public class MyParent
{
    public virtual void X()
    {
        Console.WriteLine("Executing MyParent.X");
    }
}

public class MyChild : MyParent
{
    public void DisplayOddity()
    {
        MyDelegate md = base.X;

        Console.WriteLine("Calling Invoke()");
        md.Invoke();                // Executes base method... fair enough

        Console.WriteLine("Calling BeginInvoke()");
        md.BeginInvoke(null, null); // Executes overridden method!
    }

    public override void X()
    {
        Console.WriteLine("Executing MyChild.X");
    }
}

これには、再帰呼び出しは含まれません。ただし、結果は同じ奇妙さです。

Calling Invoke()
Executing MyParent.X
Calling BeginInvoke()
Executing MyChild.X

(これがより単純な再現であることに同意する場合は、元の質問のコードを自由に置き換えてください。回答から削除します:)

正直なところ、これは私にはバグのように見えます。もう少し掘り下げてみます。

于 2008-10-23T12:27:44.880 に答える
1

Delegate.Invoke はデリゲート メソッドを直接呼び出しますが、Delegate.BeginInvoke は内部で ThreadPool.QueueUserWorkItem( ) を使用します。md.Invoke() は base.X しか呼び出せませんでした。これは、基本クラスのメソッドが base キーワードを介して派生クラス内でアクセスできるためです。スレッド プールによって開始されたデリゲートはクラスの外部にあるため、その X メソッドへの参照は、次のコードのようにオーバーロードの対象になります。



    public class Program
    {
        static void Main(string[] args)
        {
            MyChild a = new MyChild();
            MyDelegate ma = new MyDelegate(a.X);

            MyParent b = new MyChild();
            MyDelegate mb = new MyDelegate(b.X);

            ma.Invoke();
            mb.Invoke();
            ma.BeginInvoke(CallBack, null);
            mb.BeginInvoke(CallBack, null); //all four calls call derived MyChild.X

            Console.ReadLine();
        }

        public static void CallBack(IAsyncResult iAsyncResult)
        {
            return;
        }
    }

.NET Framework コードにデバッグ: http://blogs.msdn.com/sburke/archive/2008/01/16/configuring-visual-studio-to-debug-net-framework-source-code.aspx

于 2008-11-01T01:13:30.827 に答える
0

あなたが探している答えではないかもしれませんが、これはうまくいくようです:

ThreadPool.QueueUserWorkItem(x => md());

また

new Thread(() => md()).Start();

ただし、自分で会計を行う必要があります:(

于 2008-10-23T14:53:50.700 に答える