Kotlin coroutines আপনাকে পরিষ্কার, সরলীকৃত অ্যাসিঙ্ক্রোনাস কোড লিখতে সক্ষম করে যা নেটওয়ার্ক কল বা ডিস্ক অপারেশনের মতো দীর্ঘমেয়াদী কাজগুলি পরিচালনা করার সময় আপনার অ্যাপকে প্রতিক্রিয়াশীল রাখে।
এই বিষয়টি Android-এ coroutines এর বিস্তারিত বিবরণ প্রদান করে। আপনি যদি coroutines এর সাথে অপরিচিত হন, তাহলে এই বিষয় পড়ার আগে Android-এ Kotlin coroutines পড়তে ভুলবেন না।
দীর্ঘস্থায়ী কাজগুলি পরিচালনা করুন
Coroutines দীর্ঘ-চলমান কাজগুলি পরিচালনা করার জন্য দুটি অপারেশন যোগ করে নিয়মিত ফাংশন তৈরি করে। invoke (বা call ) এবং return পাশাপাশি, কোরোটিনগুলি suspend এবং resume যোগ করে:
-
suspendকরা সমস্ত স্থানীয় ভেরিয়েবল সংরক্ষণ করে বর্তমান coroutine-এর কার্য সম্পাদনে বিরতি দেয়। -
resumeযেখানে এটি স্থগিত করা হয়েছিল সেখান থেকে একটি স্থগিত coroutine কার্যকর করা অব্যাহত থাকে।
আপনি শুধুমাত্র অন্যান্য suspend ফাংশন থেকে suspend ফাংশনকে কল করতে পারেন অথবা একটি নতুন কোরোটিন শুরু করার জন্য launch মতো একটি করটিন বিল্ডার ব্যবহার করে।
নিম্নোক্ত উদাহরণটি একটি অনুমানমূলক দীর্ঘ-চলমান কাজের জন্য একটি সাধারণ কোরোটিন বাস্তবায়ন দেখায়:
suspend fun fetchDocs() { // Dispatchers.Main
val result = get("https://developer.android.com") // Dispatchers.IO for `get`
show(result) // Dispatchers.Main
}
suspend fun get(url: String) = withContext(Dispatchers.IO) { /* ... */ }
এই উদাহরণে, get() এখনও মূল থ্রেডে চলে, কিন্তু এটি নেটওয়ার্ক অনুরোধ শুরু করার আগে coroutine স্থগিত করে। নেটওয়ার্ক অনুরোধ সম্পূর্ণ হলে, মূল থ্রেডকে অবহিত করার জন্য একটি কলব্যাক ব্যবহার করার পরিবর্তে স্থগিত কোরোটিন পুনরায় শুরু get ।
কোন স্থানীয় ভেরিয়েবলের সাথে কোন ফাংশন চলছে তা পরিচালনা করতে Kotlin একটি স্ট্যাক ফ্রেম ব্যবহার করে। একটি করোটিন স্থগিত করার সময়, বর্তমান স্ট্যাক ফ্রেমটি অনুলিপি করা হয় এবং পরবর্তীতে সংরক্ষণ করা হয়। পুনরায় শুরু করার সময়, স্ট্যাক ফ্রেমটি যেখান থেকে সংরক্ষিত হয়েছিল সেখান থেকে আবার কপি করা হয় এবং ফাংশনটি আবার চলতে শুরু করে। যদিও কোডটি একটি সাধারণ ক্রমিক ব্লকিং অনুরোধের মতো দেখাতে পারে, কোরোটিন নিশ্চিত করে যে নেটওয়ার্ক অনুরোধটি মূল থ্রেডটিকে ব্লক করা এড়ায়।
প্রধান নিরাপত্তার জন্য coroutines ব্যবহার করুন
কোটলিন করোটিনগুলি প্রেরকদের ব্যবহার করে তা নির্ধারণ করতে কোন থ্রেডগুলি করোটিন সম্পাদনের জন্য ব্যবহৃত হয়। মূল থ্রেডের বাইরে কোড চালানোর জন্য, আপনি Kotlin coroutines কে বলতে পারেন ডিফল্ট বা IO ডিসপ্যাচারে কাজ করতে। কোটলিনে, সমস্ত কোরোটিন অবশ্যই একটি প্রেরণকারীতে চালাতে হবে, এমনকি যখন তারা মূল থ্রেডে চলছে। Coroutines নিজেদের স্থগিত করতে পারে, এবং প্রেরণকারী তাদের পুনরায় শুরু করার জন্য দায়ী।
কোরোটিনগুলি কোথায় চালানো উচিত তা নির্দিষ্ট করতে, কোটলিন তিনটি প্রেরণকারী সরবরাহ করে যা আপনি ব্যবহার করতে পারেন:
- Dispatchers.Main - প্রধান Android থ্রেডে একটি coroutine চালানোর জন্য এই প্রেরণকারীটি ব্যবহার করুন৷ এটি শুধুমাত্র UI এর সাথে ইন্টারঅ্যাক্ট করার জন্য এবং দ্রুত কাজ করার জন্য ব্যবহার করা উচিত। উদাহরণগুলির মধ্যে রয়েছে কল
suspendফাংশন, অ্যান্ড্রয়েড UI ফ্রেমওয়ার্ক অপারেশন চালানো এবংLiveDataঅবজেক্ট আপডেট করা। - Dispatchers.IO - এই প্রেরণকারীটি প্রধান থ্রেডের বাইরে ডিস্ক বা নেটওয়ার্ক I/O সম্পাদন করার জন্য অপ্টিমাইজ করা হয়েছে। উদাহরণগুলির মধ্যে রয়েছে রুম কম্পোনেন্ট ব্যবহার করা, ফাইল থেকে পড়া বা লেখা, এবং যেকোনো নেটওয়ার্ক অপারেশন চালানো।
- Dispatchers.Default - এই প্রেরণকারীকে প্রধান থ্রেডের বাইরে CPU-নিবিড় কাজ করার জন্য অপ্টিমাইজ করা হয়েছে। উদাহরণ ব্যবহারের ক্ষেত্রে একটি তালিকা বাছাই করা এবং JSON পার্স করা অন্তর্ভুক্ত।
পূর্ববর্তী উদাহরণটি অব্যাহত রেখে, আপনি get ফাংশন পুনরায় সংজ্ঞায়িত করতে প্রেরকদের ব্যবহার করতে পারেন। get এর বডির ভিতরে, IO থ্রেড পুলে চলে এমন একটি ব্লক তৈরি করতে withContext(Dispatchers.IO) কল করুন। যেকোন কোড আপনি যে ব্লকের ভিতরে রাখেন তা সর্বদা IO প্রেরণকারীর মাধ্যমে কার্যকর হয়। যেহেতু withContext নিজেই একটি সাসপেন্ড ফাংশন, get ফাংশনটিও একটি সাসপেন্ড ফাংশন।
suspend fun fetchDocs() { // Dispatchers.Main
val result = get("developer.android.com") // Dispatchers.Main
show(result) // Dispatchers.Main
}
suspend fun get(url: String) = // Dispatchers.Main
withContext(Dispatchers.IO) { // Dispatchers.IO (main-safety block)
/* perform network IO here */ // Dispatchers.IO (main-safety block)
} // Dispatchers.Main
}
coroutines সঙ্গে, আপনি সূক্ষ্ম-দানা নিয়ন্ত্রণ সঙ্গে থ্রেড পাঠাতে পারেন. যেহেতু withContext() আপনাকে কলব্যাক প্রবর্তন না করেই কোডের যেকোনো লাইনের থ্রেড পুল নিয়ন্ত্রণ করতে দেয়, আপনি এটিকে খুব ছোট ফাংশনে প্রয়োগ করতে পারেন যেমন একটি ডাটাবেস থেকে পড়া বা একটি নেটওয়ার্ক অনুরোধ সম্পাদন করা। প্রতিটি ফাংশন main-safe কিনা তা নিশ্চিত করার জন্য withContext() ব্যবহার করা একটি ভাল অভ্যাস, যার মানে হল আপনি প্রধান থ্রেড থেকে ফাংশনটি কল করতে পারেন। এইভাবে, কলারকে কখনই ভাবতে হবে না যে ফাংশনটি কার্যকর করতে কোন থ্রেড ব্যবহার করা উচিত।
পূর্ববর্তী উদাহরণে, fetchDocs() প্রধান থ্রেডে কার্যকর করে; যাইহোক, এটি নিরাপদে get কল করতে পারে, যা পটভূমিতে একটি নেটওয়ার্ক অনুরোধ সম্পাদন করে। যেহেতু coroutines suspend এবং resume সমর্থন করে, তাই withContext ব্লকটি সম্পন্ন হওয়ার সাথে সাথেই মূল থ্রেডে coroutine পুনরায় শুরু get হয়।
withContext() এর কর্মক্ষমতা
withContext() সমতুল্য কলব্যাক-ভিত্তিক বাস্তবায়নের তুলনায় অতিরিক্ত ওভারহেড যোগ করে না। তদ্ব্যতীত, কিছু পরিস্থিতিতে সমতুল্য কলব্যাক-ভিত্তিক বাস্তবায়নের বাইরে withContext() অপ্টিমাইজ করা সম্ভব। উদাহরণস্বরূপ, যদি একটি ফাংশন একটি নেটওয়ার্কে দশটি কল করে, আপনি Kotlin কে বলতে পারেন একটি outer withContext() ব্যবহার করে শুধুমাত্র একবার থ্রেডগুলি পরিবর্তন করতে। তারপরে, যদিও নেটওয়ার্ক লাইব্রেরি withContext() একাধিকবার ব্যবহার করে, এটি একই প্রেরণকারীতে থাকে এবং থ্রেড পরিবর্তন করা এড়িয়ে যায়। উপরন্তু, Kotlin যখনই সম্ভব থ্রেড সুইচ এড়াতে Dispatchers.Default এবং Dispatchers.IO এর মধ্যে সুইচিং অপ্টিমাইজ করে।
একটি করুটিন শুরু করুন
আপনি দুটি উপায়ের একটিতে কোরোটিন শুরু করতে পারেন:
-
launchএকটি নতুন কোরোটিন শুরু করে এবং কলকারীকে ফলাফল ফেরত দেয় না। "ফায়ার অ্যান্ড বিফোর" হিসাবে বিবেচিত যে কোনও কাজlaunchব্যবহার করে শুরু করা যেতে পারে। -
asyncএকটি নতুন coroutine শুরু করে এবং আপনাকেawaitনামক একটি সাসপেন্ড ফাংশন সহ ফলাফল ফেরত দিতে দেয়।
সাধারণত, আপনার একটি নিয়মিত ফাংশন থেকে একটি নতুন কোরোটিন launch উচিত, কারণ একটি নিয়মিত ফাংশন await করতে পারে না। async ব্যবহার করুন শুধুমাত্র যখন অন্য কোরোটিনের ভিতরে বা যখন একটি সাসপেন্ড ফাংশনের ভিতরে এবং সমান্তরাল পচন সম্পাদন করে।
সমান্তরাল পচন
একটি suspend ফাংশনের ভিতরে শুরু হওয়া সমস্ত কোরোটিনগুলি যখন সেই ফাংশনটি ফিরে আসে তখন অবশ্যই বন্ধ করতে হবে, তাই আপনাকে অবশ্যই গ্যারান্টি দিতে হবে যে সেই কোরোটিনগুলি ফিরে আসার আগে শেষ হবে৷ Kotlin-এ স্ট্রাকচার্ড কনকারেন্সি দিয়ে, আপনি একটি coroutineScope সংজ্ঞায়িত করতে পারেন যা এক বা একাধিক coroutine শুরু করে। তারপর, await() (একটি coroutine এর জন্য) বা awaitAll() (একাধিক কোরোটিনের জন্য) ব্যবহার করে, আপনি গ্যারান্টি দিতে পারেন যে ফাংশন থেকে ফিরে আসার আগে এই coroutines শেষ হয়ে যাবে।
একটি উদাহরণ হিসাবে, আসুন একটি coroutineScope সংজ্ঞায়িত করা যাক যা অ্যাসিঙ্ক্রোনাসভাবে দুটি নথি নিয়ে আসে। প্রতিটি বিলম্বিত রেফারেন্সে await() কল করার মাধ্যমে, আমরা গ্যারান্টি দিই যে একটি মান ফেরত দেওয়ার আগে উভয় async অপারেশন শেষ:
suspend fun fetchTwoDocs() =
coroutineScope {
val deferredOne = async { fetchDoc(1) }
val deferredTwo = async { fetchDoc(2) }
deferredOne.await()
deferredTwo.await()
}
আপনি সংগ্রহে awaitAll() ব্যবহার করতে পারেন, যেমনটি নিম্নলিখিত উদাহরণে দেখানো হয়েছে:
suspend fun fetchTwoDocs() = // called on any Dispatcher (any thread, possibly Main)
coroutineScope {
val deferreds = listOf( // fetch two docs at the same time
async { fetchDoc(1) }, // async returns a result for the first doc
async { fetchDoc(2) } // async returns a result for the second doc
)
deferreds.awaitAll() // use awaitAll to wait for both network requests
}
যদিও fetchTwoDocs() async এর সাথে নতুন coroutines লঞ্চ করে, ফাংশনটি awaitAll() ব্যবহার করে সেই লঞ্চ করা কোরোটিনগুলি ফিরে আসার আগে শেষ হওয়ার জন্য অপেক্ষা করে। মনে রাখবেন যে, যদিও আমরা awaitAll() কে কল না করে থাকি, coroutineScope বিল্ডার সমস্ত নতুন coroutines সম্পূর্ণ না হওয়া পর্যন্ত fetchTwoDocs নামক coroutine পুনরায় শুরু করে না।
উপরন্তু, coroutineScope যেকোন ব্যতিক্রম ক্যাচ করে যেগুলি coroutines ছুড়ে দেয় এবং সেগুলিকে কলারের কাছে ফিরিয়ে দেয়।
সমান্তরাল পচন সম্পর্কে আরও তথ্যের জন্য, সাসপেন্ডিং ফাংশন রচনা করা দেখুন।
Coroutines ধারণা
CoroutineScope
একটি CoroutineScope launch বা async ব্যবহার করে তৈরি করা যেকোনো কোরোটিনের ট্র্যাক রাখে। চলমান কাজ (অর্থাৎ চলমান coroutines) যেকোনো সময়ে scope.cancel() কল করে বাতিল করা যেতে পারে। অ্যান্ড্রয়েডে, কিছু KTX লাইব্রেরি নির্দিষ্ট জীবনচক্র ক্লাসের জন্য তাদের নিজস্ব CoroutineScope প্রদান করে। উদাহরণস্বরূপ, ViewModel একটি viewModelScope আছে এবং Lifecycle lifecycleScope আছে। একটি প্রেরক থেকে ভিন্ন, যাইহোক, একটি CoroutineScope coroutines চালায় না।
Android-এ Coroutines-এর সাথে ব্যাকগ্রাউন্ড থ্রেডিং -এ পাওয়া উদাহরণগুলিতে viewModelScope ব্যবহার করা হয়। যাইহোক, যদি আপনার অ্যাপের একটি নির্দিষ্ট স্তরে কোরোটিনের জীবনচক্র নিয়ন্ত্রণ করতে আপনার নিজস্ব CoroutineScope তৈরি করতে হয়, তাহলে আপনি নিম্নরূপ একটি তৈরি করতে পারেন:
class ExampleClass {
// Job and Dispatcher are combined into a CoroutineContext which
// will be discussed shortly
val scope = CoroutineScope(Job() + Dispatchers.Main)
fun exampleMethod() {
// Starts a new coroutine within the scope
scope.launch {
// New coroutine that can call suspend functions
fetchDocs()
}
}
fun cleanUp() {
// Cancel the scope to cancel ongoing coroutines work
scope.cancel()
}
}
একটি বাতিল সুযোগ আরো coroutines তৈরি করতে পারে না. অতএব, আপনার scope.cancel() কল করা উচিৎ তখনই যখন তার জীবনচক্র নিয়ন্ত্রণকারী ক্লাসটি ধ্বংস হয়ে যাচ্ছে। viewModelScope ব্যবহার করার সময়, ViewModel ক্লাস ViewModel এর onCleared() পদ্ধতিতে আপনার জন্য স্বয়ংক্রিয়ভাবে সুযোগ বাতিল করে।
চাকরি
একটি Job হল একটি করুটিনের একটি হাতল৷ আপনি launch বা async মাধ্যমে তৈরি করা প্রতিটি কোরোটিন একটি Job উদাহরণ প্রদান করে যা অনন্যভাবে করটিনকে সনাক্ত করে এবং এর জীবনচক্র পরিচালনা করে। আপনি একটি Job একটি CoroutineScope এর জীবনচক্রকে আরও পরিচালনা করতে পাস করতে পারেন, যেমনটি নিম্নলিখিত উদাহরণে দেখানো হয়েছে:
class ExampleClass {
...
fun exampleMethod() {
// Handle to the coroutine, you can control its lifecycle
val job = scope.launch {
// New coroutine
}
if (...) {
// Cancel the coroutine started above, this doesn't affect the scope
// this coroutine was launched in
job.cancel()
}
}
}
Coroutine Context
একটি CoroutineContext নিম্নলিখিত উপাদানগুলির সেট ব্যবহার করে একটি করটিনের আচরণকে সংজ্ঞায়িত করে:
-
Job: করোটিনের জীবনচক্র নিয়ন্ত্রণ করে। -
CoroutineDispatcher: ডিসপ্যাচগুলি উপযুক্ত থ্রেডে কাজ করে। -
CoroutineName: coroutine এর নাম, ডিবাগিংয়ের জন্য দরকারী। -
CoroutineExceptionHandler: ধরা না পড়া ব্যতিক্রমগুলি পরিচালনা করে।
একটি সুযোগের মধ্যে তৈরি করা নতুন coroutines-এর জন্য, নতুন coroutine-এ একটি নতুন Job উদাহরণ বরাদ্দ করা হয়, এবং অন্যান্য CoroutineContext উপাদানগুলি অন্তর্ভুক্ত সুযোগ থেকে উত্তরাধিকার সূত্রে প্রাপ্ত হয়। আপনি launch বা async ফাংশনে একটি নতুন CoroutineContext পাস করে উত্তরাধিকার সূত্রে প্রাপ্ত উপাদানগুলিকে ওভাররাইড করতে পারেন। মনে রাখবেন যে একটি Job launch বা async জন্য কোনও প্রভাব নেই, কারণ Job একটি নতুন উদাহরণ সর্বদা একটি নতুন কোরোটিনে বরাদ্দ করা হয়।
class ExampleClass {
val scope = CoroutineScope(Job() + Dispatchers.Main)
fun exampleMethod() {
// Starts a new coroutine on Dispatchers.Main as it's the scope's default
val job1 = scope.launch {
// New coroutine with CoroutineName = "coroutine" (default)
}
// Starts a new coroutine on Dispatchers.Default
val job2 = scope.launch(Dispatchers.Default + CoroutineName("BackgroundCoroutine")) {
// New coroutine with CoroutineName = "BackgroundCoroutine" (overridden)
}
}
}
অতিরিক্ত coroutines সম্পদ
আরও কোরোটিন সংস্থানগুলির জন্য, নিম্নলিখিত লিঙ্কগুলি দেখুন: