আপনার কোড আপনাকে যা ভাবেন তা করতে পারে না। অ্যাসিঙ্ক পদ্ধতিগুলি অ্যাসিঙ্ক ফলাফলের জন্য অপেক্ষা করা শুরু করার সাথে সাথেই ফিরে আসে। কোডটি আসলে কীভাবে আচরণ করছে তা খতিয়ে দেখতে ট্রেসিং ব্যবহার করা অন্তর্দৃষ্টিপূর্ণ।
নীচের কোডটি নিম্নলিখিতটি করে:
- 4 টি কার্য তৈরি করুন
- প্রতিটি টাস্ক অবিচ্ছিন্নভাবে একটি সংখ্যা বাড়িয়ে তুলবে এবং বর্ধিত সংখ্যাটি ফিরিয়ে দেবে
- অ্যাসিঙ্ক ফলাফল এলে এটি সনাক্ত করা হয়।
static TypeHashes _type = new TypeHashes(typeof(Program));
private void Run()
{
TracerConfig.Reset("debugoutput");
using (Tracer t = new Tracer(_type, "Run"))
{
for (int i = 0; i < 4; i++)
{
DoSomeThingAsync(i);
}
}
Application.Run(); // Start window message pump to prevent termination
}
private async void DoSomeThingAsync(int i)
{
using (Tracer t = new Tracer(_type, "DoSomeThingAsync"))
{
t.Info("Hi in DoSomething {0}",i);
try
{
int result = await Calculate(i);
t.Info("Got async result: {0}", result);
}
catch (ArgumentException ex)
{
t.Error("Got argument exception: {0}", ex);
}
}
}
Task<int> Calculate(int i)
{
var t = new Task<int>(() =>
{
using (Tracer t2 = new Tracer(_type, "Calculate"))
{
if( i % 2 == 0 )
throw new ArgumentException(String.Format("Even argument {0}", i));
return i++;
}
});
t.Start();
return t;
}
আপনি ট্রেস পর্যবেক্ষণ যখন
22:25:12.649 02172/02820 { AsyncTest.Program.Run
22:25:12.656 02172/02820 { AsyncTest.Program.DoSomeThingAsync
22:25:12.657 02172/02820 Information AsyncTest.Program.DoSomeThingAsync Hi in DoSomething 0
22:25:12.658 02172/05220 { AsyncTest.Program.Calculate
22:25:12.659 02172/02820 { AsyncTest.Program.DoSomeThingAsync
22:25:12.659 02172/02820 Information AsyncTest.Program.DoSomeThingAsync Hi in DoSomething 1
22:25:12.660 02172/02756 { AsyncTest.Program.Calculate
22:25:12.662 02172/02820 { AsyncTest.Program.DoSomeThingAsync
22:25:12.662 02172/02820 Information AsyncTest.Program.DoSomeThingAsync Hi in DoSomething 2
22:25:12.662 02172/02820 { AsyncTest.Program.DoSomeThingAsync
22:25:12.662 02172/02820 Information AsyncTest.Program.DoSomeThingAsync Hi in DoSomething 3
22:25:12.664 02172/02756 } AsyncTest.Program.Calculate Duration 4ms
22:25:12.666 02172/02820 } AsyncTest.Program.Run Duration 17ms ---- Run has completed. The async methods are now scheduled on different threads.
22:25:12.667 02172/02756 Information AsyncTest.Program.DoSomeThingAsync Got async result: 1
22:25:12.667 02172/02756 } AsyncTest.Program.DoSomeThingAsync Duration 8ms
22:25:12.667 02172/02756 { AsyncTest.Program.Calculate
22:25:12.665 02172/05220 Exception AsyncTest.Program.Calculate Exception thrown: System.ArgumentException: Even argument 0
at AsyncTest.Program.c__DisplayClassf.Calculateb__e() in C:\Source\AsyncTest\AsyncTest\Program.cs:line 124
at System.Threading.Tasks.Task`1.InvokeFuture(Object futureAsObj)
at System.Threading.Tasks.Task.InnerInvoke()
at System.Threading.Tasks.Task.Execute()
22:25:12.668 02172/02756 Exception AsyncTest.Program.Calculate Exception thrown: System.ArgumentException: Even argument 2
at AsyncTest.Program.c__DisplayClassf.Calculateb__e() in C:\Source\AsyncTest\AsyncTest\Program.cs:line 124
at System.Threading.Tasks.Task`1.InvokeFuture(Object futureAsObj)
at System.Threading.Tasks.Task.InnerInvoke()
at System.Threading.Tasks.Task.Execute()
22:25:12.724 02172/05220 } AsyncTest.Program.Calculate Duration 66ms
22:25:12.724 02172/02756 } AsyncTest.Program.Calculate Duration 57ms
22:25:12.725 02172/05220 Error AsyncTest.Program.DoSomeThingAsync Got argument exception: System.ArgumentException: Even argument 0
Server stack trace:
at AsyncTest.Program.c__DisplayClassf.Calculateb__e() in C:\Source\AsyncTest\AsyncTest\Program.cs:line 124
at System.Threading.Tasks.Task`1.InvokeFuture(Object futureAsObj)
at System.Threading.Tasks.Task.InnerInvoke()
at System.Threading.Tasks.Task.Execute()
Exception rethrown at [0]:
at System.Runtime.CompilerServices.TaskAwaiter.EndAwait()
at System.Runtime.CompilerServices.TaskAwaiter`1.EndAwait()
at AsyncTest.Program.DoSomeThingAsyncd__8.MoveNext() in C:\Source\AsyncTest\AsyncTest\Program.cs:line 106
22:25:12.725 02172/02756 Error AsyncTest.Program.DoSomeThingAsync Got argument exception: System.ArgumentException: Even argument 2
Server stack trace:
at AsyncTest.Program.c__DisplayClassf.Calculateb__e() in C:\Source\AsyncTest\AsyncTest\Program.cs:line 124
at System.Threading.Tasks.Task`1.InvokeFuture(Object futureAsObj)
at System.Threading.Tasks.Task.InnerInvoke()
at System.Threading.Tasks.Task.Execute()
Exception rethrown at [0]:
at System.Runtime.CompilerServices.TaskAwaiter.EndAwait()
at System.Runtime.CompilerServices.TaskAwaiter`1.EndAwait()
at AsyncTest.Program.DoSomeThingAsyncd__8.MoveNext() in C:\Source\AsyncTest\AsyncTest\Program.cs:line 0
22:25:12.726 02172/05220 } AsyncTest.Program.DoSomeThingAsync Duration 70ms
22:25:12.726 02172/02756 } AsyncTest.Program.DoSomeThingAsync Duration 64ms
22:25:12.726 02172/05220 { AsyncTest.Program.Calculate
22:25:12.726 02172/05220 } AsyncTest.Program.Calculate Duration 0ms
22:25:12.726 02172/05220 Information AsyncTest.Program.DoSomeThingAsync Got async result: 3
22:25:12.726 02172/05220 } AsyncTest.Program.DoSomeThingAsync Duration 64ms
আপনি লক্ষ্য করবেন যে রান পদ্ধতিটি 2820 থ্রেডে সম্পূর্ণ হয়েছে যখন কেবলমাত্র একটি সন্তানের থ্রেড শেষ হয়েছে (2756)। আপনি যদি আপনার প্রতীক্ষিত পদ্ধতির আশেপাশে একটি চেষ্টা / ক্যাপচার রাখেন তবে আপনি ব্যতিক্রমটিকে স্বাভাবিক উপায়ে "ধরতে" পারেন যদিও আপনার কোডটি অন্য থ্রেডে কার্যকর করা হয় যখন গণনার টাস্কটি শেষ হয়ে যায় এবং আপনার ধারণাটি কার্যকর করা হয় uted
গণনা পদ্ধতিটি নিক্ষিপ্ত ব্যতিক্রমগুলি স্বয়ংক্রিয়ভাবে ট্রেস করে কারণ আমি অ্যাপি চ্যাঞ্জ থেকে ApiChange.Api.dll ব্যবহার করেছি because সরঞ্জামটি ব্যবহার করেছি। ট্র্যাকিং এবং রিফ্লেক্টর যা চলছে তা বুঝতে অনেক সাহায্য করে। থ্রেডিং থেকে পরিত্রাণ পেতে আপনি নিজের গেটএইটার বিগেইনওয়েট এবং এন্ডএইয়েটের নিজস্ব সংস্করণ তৈরি করতে পারেন এবং কোনও টাস্ক নয় তবে উদাহরণস্বরূপ আপনার নিজস্ব এক্সটেনশন পদ্ধতির মধ্যে একটি অলস এবং ট্রেস তৈরি করতে পারেন। তারপরে আপনি আরও ভালভাবে বুঝতে পারবেন যে সংকলকটি কি করে এবং টিপিএল কী করে।
এখন আপনি দেখতে পাচ্ছেন যে আপনার ব্যতিক্রমটিকে ফিরিয়ে আনার / চেষ্টা করার কোনও উপায় নেই কারণ প্রচারের কোনও ব্যতিক্রমের জন্য কোনও স্ট্যাক ফ্রেম নেই। আপনি async ক্রিয়াকলাপ শুরু করার পরে আপনার কোড সম্পূর্ণ ভিন্ন কিছু করছে। এটি থ্রেড কল করতে পারে leep ঘুম বা এমনকি সমাপ্ত। যতক্ষণ না কোনও পূর্বভূমি থ্রেড থাকে ততক্ষণ আপনার অ্যাপ্লিকেশন আনন্দের সাথে অ্যাসিক্রোনাস কাজগুলি চালিয়ে যেতে থাকবে।
আপনার অ্যাসিনক্রোনাস অপারেশন শেষ হয়ে গেলে এবং ইউআই থ্রেডে ফিরে কল করার পরে আপনি অ্যাসিঙ্ক পদ্ধতির অভ্যন্তরে ব্যতিক্রমটি পরিচালনা করতে পারেন। এটি করার প্রস্তাবিত উপায়টি হল TaskScheduler.FromSynchronizationContext এর সাথে । এটি কেবল তখনই কাজ করে যদি আপনার কাছে একটি ইউআই থ্রেড থাকে এবং এটি অন্যান্য জিনিসে খুব বেশি ব্যস্ত না হয়।