2

チェックポイントを短縮する方法はありますか?BackgroundWorker.CancellationPending

たとえば、return以下のサンプルコードのようにカプセル化する方法はありますか?:

//REAL CODE (CURRENTLY USE THIS)
if (this.TW.CancellationPending) 
    return; 

//PSEUDO REPLACEMENT CODE
this.CkPt(CurrentMethod); //PSEUDO USAGE 
    //^^^ PARAMETER IS A REFERENCE TO THE CURRENT METHOD, SIMILAR TO `this` FOR AN OBJECT  
//OR MAYBE AN EXTENSION METHOD WOULD LOOK CLEANER
CurrentMethod.CkPt(); //PSEUDO USAGE

private void CkPt(Method m) //PSEUDO METHOD
{
    /*
        POSSIBLY PERFORM OTHER CHECKPOINT TASKS HERE
    */
    if (this.TW.CancellationPending) 
        m.return/*FROM METHOD THAT CALLED ME*/;
}  

私はこのようなマルチチェックポイントの状況をより読みやすくしようとしています:

//PSUEDO METHOD 
//DO NOT TAKE THIS AS REPEATING CODE
//IT IS ONLY MEANT TO SHOW MULTIPLE USES OF THE SAME CHECKPOINT
//MY REAL TASK METHOD(S) CONTAIN MANY MANY MANY AREAS THAT DON'T REPEAT AND REQUIRE CHECKPOINTS  
public void FakeBWTask()
{
    if (this.TW.CancellationPending) 
        return; 

    foreach (var F1 in Fake1)
    {
        if (this.TW.CancellationPending) 
            return; 

        foreach (var F2 in Fake2)
        {   
            if (this.TW.CancellationPending) 
                return; 
            foreach (var F3 in Fake3)
            {
                if (this.TW.CancellationPending) 
                    return; 
            }
        }
    }
}

助けてくれてありがとう!

4

2 に答える 2

5

methodAを呼び出してmethodBを呼び出し、methodBにmethodAを返す方法はありません(末尾再帰がないため)

このようなイテレータの使用を検討してください。チェックポイントの間にtry/catchを置くことができるいくつかの状況で機能します。

public void FakeBWTask()
{
   if (this.TW.CancellationPending) 
      return; 
   foreach (object ignore in FakeBWTaskSteps())
   {
      // Other checkpoint logic here....
      if (this.TW.CancellationPending) 
          return; 
   }
}

private IEnumerable<object> FakeBWTaskSteps()
{
   Part1();
   yield return null; // Execute checkpoint logic.

   Part2();
   yield return null; // Execute checkpoint logic.

   Part3();
   yield return null; // Execute checkpoint logic.

   Part4();
   yield return null; // Execute checkpoint logic.

   // Do some other stuff.
   yield return null; // Execute checkpoint logic.

   // final stuff.  No looping here.
}
于 2011-08-03T23:03:21.967 に答える
1

mono.Cecilを使用してILの書き換えを行うことができます。「Cancellable」属性を使用してメソッドを装飾してから、その属性を使用してメソッドを書き換えることができます。命令ストリームからCFGを構築し、スタック深度を計算するコードを記述し、スタック深度がゼロの場所を特定し、ノードを挿入してキャンセルチェックを実行してから、CFGをメソッド本体に再シリアル化する必要があります。おそらく約1週間かかり、さらにすべてをテストするのに時間がかかります。また、ビルド後のステップを追加する必要があります。

一般に、キャンセル可能なメソッドがたくさんない限り、おそらくしばらくの間は価値がありません。

于 2011-08-04T01:38:48.760 に答える