以下の要件でバックグラウンド非同期のファイル読み込みをしたいと考えています。
そこでproducer-consumerパターンで実装してみたのですが上手く行かずつまづいています。
・読み込みタスクはずっと稼働し続ける。
・読み込みタスクのオーダーは途中でキャンセルされる場合がある。
(その時あったリクエストは全て破棄)
・キャンセル後は再びオーダー待ちに戻る。
・オーダーはいつ入るか分からない。
・読み込み終わったらその段階でUIスレッドに結果を渡す。
オーダーが不定期なため、CPUリソースを抑えるためにBlockingCollection<T>を使用しています。
しかし逆にこれがキャンセルとの相性を悪くしており、読み込みタスク制御が上手く行きません。
Cancel()でタスクを止めたいのですが、BlockingCollectionが文字通りブロッキングしている時があるため、キャンセルをかけても受け付けない状態(WatingForActivattion)が出てきます。
orderをCompleteAdding()かけて動かそうとしても、今度はかかるタイミングでUIのデッドロックが発生してしまい、結局止められないケースが出ています。
BlockingCollectionを使用しなければ比較的簡単に解決できると思いますが、ずっとCPUリソースを消費してしまうため最終手段としたいと思っています。
非同期プログラミングに不慣れなので、良い実装方法を探しています。
いい解決法などありましたらよろしくおねがいします。
以下に簡易ソースを書きます。
C#
1//UIスレッドにデータを返すためのインターフェイス 2public interface IFileDataSetter 3{ 4 Task SetFileDataAsync(byte[] data); 5} 6 7//読み込みタスク側 8public class AsyncLoader 9{ 10 private IFileDataSetter setter; //実装先はForm 11 private BlockingCollection<string> order = new BlokingCollection<string>(); 12 private CancellationTokenSource cts = new CancellationTokenSource(); 13 private Task loadTask; 14 15 public void LoadStart() 16 { 17 loadTask = loadBackground(); 18 } 19 20 private Task loadBackground() 21 { 22 return Task.Run(() => 23 { 24 foreach (var fileName in order.GetConsumingEnumerable()) 25 { 26 //ファイルを読み込みUIスレッドに結果を渡す 27 await setter.SetFileDataAsync(data); 28 } 29 }, cts.Token); 30 } 31 32 public void Order(string fileName) 33 { 34 order.Add(fileName); 35 } 36 37 public void Cancel() 38 { 39 cts.Cancel(true); //読み込みタスクをキャンセル 40 while (!loadTask.IsCanceled) //←ブロッキングされてる時はずっとキャンセルできない 41 { 42 Thread.Sleep(100); 43 } 44 loadTask = loadBackground(); 45 } 46}

回答1件
あなたの回答
tips
プレビュー
バッドをするには、ログインかつ
こちらの条件を満たす必要があります。
退会済みユーザー
2015/10/22 07:02
2015/10/22 07:30
退会済みユーザー
2015/10/22 08:36