একটি কোরাউটিন হল একটি কনকারেন্সি ডিজাইন প্যাটার্ন যা আপনি অ্যাসিঙ্ক্রোনাসভাবে কার্যকর করা কোডকে সহজ করতে Android এ ব্যবহার করতে পারেন। Coroutines সংস্করণ 1.3 এ Kotlin এ যোগ করা হয়েছে এবং অন্যান্য ভাষার প্রতিষ্ঠিত ধারণার উপর ভিত্তি করে তৈরি করা হয়েছে।
অ্যান্ড্রয়েডে, কোরোটিনগুলি দীর্ঘমেয়াদী কাজগুলি পরিচালনা করতে সাহায্য করে যা অন্যথায় মূল থ্রেডটিকে ব্লক করতে পারে এবং আপনার অ্যাপটিকে প্রতিক্রিয়াহীন হয়ে পড়তে পারে। 50% এরও বেশি পেশাদার বিকাশকারীরা যারা কোরোটিন ব্যবহার করেন তারা উত্পাদনশীলতা বৃদ্ধির কথা জানিয়েছেন। এই বিষয়গুলি বর্ণনা করে যে আপনি কীভাবে এই সমস্যাগুলি সমাধান করতে Kotlin coroutines ব্যবহার করতে পারেন, আপনাকে ক্লিনার এবং আরও সংক্ষিপ্ত অ্যাপ কোড লিখতে সক্ষম করে৷
বৈশিষ্ট্য
Coroutines হল Android এ অ্যাসিঙ্ক্রোনাস প্রোগ্রামিংয়ের জন্য আমাদের প্রস্তাবিত সমাধান। উল্লেখযোগ্য বৈশিষ্ট্যগুলির মধ্যে নিম্নলিখিতগুলি অন্তর্ভুক্ত রয়েছে:
- লাইটওয়েট : সাসপেনশনের জন্য সমর্থনের কারণে আপনি একটি একক থ্রেডে অনেকগুলি কোরোটিন চালাতে পারেন, যা থ্রেডটি যেখানে কোরোটিন চলছে সেখানে ব্লক করে না। সাসপেন্ড করা অনেক সমসাময়িক ক্রিয়াকলাপকে সমর্থন করার সময় ব্লক করার সময় মেমরি সংরক্ষণ করে।
- কম মেমরি ফাঁস : একটি সুযোগের মধ্যে ক্রিয়াকলাপ চালানোর জন্য স্ট্রাকচার্ড কনকারেন্সি ব্যবহার করুন।
- অন্তর্নির্মিত বাতিলকরণ সমর্থন : চলমান করোটিন শ্রেণিবিন্যাসের মাধ্যমে বাতিলকরণ স্বয়ংক্রিয়ভাবে প্রচারিত হয়।
- জেটপ্যাক ইন্টিগ্রেশন : অনেক জেটপ্যাক লাইব্রেরিতে এমন এক্সটেনশন রয়েছে যা সম্পূর্ণ কোরোটিন সমর্থন প্রদান করে। কিছু লাইব্রেরি তাদের নিজস্ব কোরোটিন স্কোপও প্রদান করে যা আপনি স্ট্রাকচার্ড কনকারেন্সির জন্য ব্যবহার করতে পারেন।
উদাহরণ ওভারভিউ
অ্যাপ আর্কিটেকচারের গাইডের উপর ভিত্তি করে, এই বিষয়ের উদাহরণগুলি একটি নেটওয়ার্ক অনুরোধ করে এবং ফলাফলটিকে মূল থ্রেডে ফেরত দেয়, যেখানে অ্যাপটি ব্যবহারকারীর কাছে ফলাফলটি প্রদর্শন করতে পারে।
বিশেষত, ViewModel
আর্কিটেকচার উপাদান নেটওয়ার্ক অনুরোধটি ট্রিগার করতে মূল থ্রেডের সংগ্রহস্থল স্তরটিকে কল করে। এই নির্দেশিকাটি বিভিন্ন সমাধানের মাধ্যমে পুনরাবৃত্তি করে যা মূল থ্রেড আনব্লক রাখতে coroutines ব্যবহার করে।
ViewModel
KTX এক্সটেনশনের একটি সেট রয়েছে যা সরাসরি কোরোটিনের সাথে কাজ করে। এই এক্সটেনশনগুলি হল lifecycle-viewmodel-ktx
লাইব্রেরি এবং এই গাইডে ব্যবহার করা হয়েছে৷
নির্ভরতা তথ্য
আপনার অ্যান্ড্রয়েড প্রোজেক্টে কোরোটিন ব্যবহার করতে, আপনার অ্যাপের build.gradle
ফাইলে নিম্নলিখিত নির্ভরতা যোগ করুন:
গ্রোভি
dependencies { implementation 'org.jetbrains.kotlinx:kotlinx-coroutines-android:1.3.9' }
কোটলিন
dependencies { implementation("org.jetbrains.kotlinx:kotlinx-coroutines-android:1.3.9") }
একটি ব্যাকগ্রাউন্ড থ্রেডে কার্যকর করা হচ্ছে
প্রধান থ্রেডে একটি নেটওয়ার্ক অনুরোধ করা এটি একটি প্রতিক্রিয়া না পাওয়া পর্যন্ত অপেক্ষা করতে বা ব্লক করে দেয়। যেহেতু থ্রেডটি ব্লক করা আছে, তাই OS onDraw()
কল করতে পারে না, যার ফলে আপনার অ্যাপ ফ্রিজ হয়ে যায় এবং সম্ভাব্যভাবে একটি অ্যাপ্লিকেশন নট রেসপন্ডিং (ANR) ডায়ালগের দিকে নিয়ে যায়। একটি ভাল ব্যবহারকারীর অভিজ্ঞতার জন্য, আসুন একটি ব্যাকগ্রাউন্ড থ্রেডে এই অপারেশনটি চালাই।
প্রথমে, আসুন আমাদের Repository
ক্লাসটি একবার দেখে নেওয়া যাক এবং দেখুন কিভাবে এটি নেটওয়ার্ক অনুরোধ করছে:
sealed class Result<out R> {
data class Success<out T>(val data: T) : Result<T>()
data class Error(val exception: Exception) : Result<Nothing>()
}
class LoginRepository(private val responseParser: LoginResponseParser) {
private const val loginUrl = "https://example.com/login"
// Function that makes the network request, blocking the current thread
fun makeLoginRequest(
jsonBody: String
): Result<LoginResponse> {
val url = URL(loginUrl)
(url.openConnection() as? HttpURLConnection)?.run {
requestMethod = "POST"
setRequestProperty("Content-Type", "application/json; utf-8")
setRequestProperty("Accept", "application/json")
doOutput = true
outputStream.write(jsonBody.toByteArray())
return Result.Success(responseParser.parse(inputStream))
}
return Result.Error(Exception("Cannot open HttpURLConnection"))
}
}
makeLoginRequest
সিঙ্ক্রোনাস এবং কলিং থ্রেড ব্লক করে। নেটওয়ার্ক অনুরোধের প্রতিক্রিয়া মডেল করার জন্য, আমাদের নিজস্ব Result
ক্লাস আছে।
ViewModel
নেটওয়ার্ক অনুরোধ ট্রিগার করে যখন ব্যবহারকারী ক্লিক করে, উদাহরণস্বরূপ, একটি বোতামে:
class LoginViewModel(
private val loginRepository: LoginRepository
): ViewModel() {
fun login(username: String, token: String) {
val jsonBody = "{ username: \"$username\", token: \"$token\"}"
loginRepository.makeLoginRequest(jsonBody)
}
}
পূর্ববর্তী কোডের সাথে, নেটওয়ার্ক অনুরোধ করার সময় LoginViewModel
UI থ্রেড ব্লক করছে। মূল থ্রেড থেকে এক্সিকিউশন সরানোর সবচেয়ে সহজ সমাধান হল একটি নতুন করুটিন তৈরি করা এবং একটি I/O থ্রেডে নেটওয়ার্ক অনুরোধ চালানো:
class LoginViewModel(
private val loginRepository: LoginRepository
): ViewModel() {
fun login(username: String, token: String) {
// Create a new coroutine to move the execution off the UI thread
viewModelScope.launch(Dispatchers.IO) {
val jsonBody = "{ username: \"$username\", token: \"$token\"}"
loginRepository.makeLoginRequest(jsonBody)
}
}
}
চলুন login
ফাংশনে coroutines কোড বিচ্ছিন্ন করা যাক:
-
viewModelScope
হল একটি পূর্বনির্ধারিতCoroutineScope
যাViewModel
KTX এক্সটেনশনের সাথে অন্তর্ভুক্ত। মনে রাখবেন যে সমস্ত কোরোটিন অবশ্যই একটি সুযোগে চলবে। একটিCoroutineScope
এক বা একাধিক সম্পর্কিত coroutines পরিচালনা করে। -
launch
হল একটি ফাংশন যা একটি করুটিন তৈরি করে এবং এর ফাংশন বডির এক্সিকিউশনকে সংশ্লিষ্ট প্রেরককে প্রেরণ করে। -
Dispatchers.IO
নির্দেশ করে যে I/O অপারেশনের জন্য সংরক্ষিত একটি থ্রেডে এই করুটিনটি কার্যকর করা উচিত।
login
ফাংশনটি নিম্নরূপ সঞ্চালিত হয়:
- অ্যাপটি মূল থ্রেডের
View
লেয়ার থেকেlogin
ফাংশনকে কল করে। -
launch
একটি নতুন কোরোটিন তৈরি করে, এবং নেটওয়ার্ক অনুরোধটি I/O অপারেশনের জন্য সংরক্ষিত একটি থ্রেডে স্বাধীনভাবে করা হয়। - করুটিন চলাকালীন,
login
ফাংশনটি কার্যকর করা চালিয়ে যায় এবং রিটার্ন করে, সম্ভবত নেটওয়ার্ক অনুরোধ শেষ হওয়ার আগে। মনে রাখবেন যে সরলতার জন্য, নেটওয়ার্ক প্রতিক্রিয়া আপাতত উপেক্ষা করা হয়েছে।
যেহেতু এই coroutineটি viewModelScope
দিয়ে শুরু হয়েছে, তাই এটি ViewModel
এর সুযোগে কার্যকর করা হয়। ব্যবহারকারী স্ক্রীন থেকে দূরে নেভিগেট করার কারণে যদি ViewModel
টি ধ্বংস হয়ে যায়, তাহলে viewModelScope
স্বয়ংক্রিয়ভাবে বাতিল হয়ে যাবে এবং চলমান সমস্ত কোরোটিনও বাতিল হয়ে যাবে।
পূর্ববর্তী উদাহরণের সাথে একটি সমস্যা হল যে makeLoginRequest
কল করা যেকোন কিছুকে স্পষ্টভাবে মূল থ্রেড থেকে এক্সিকিউশন সরানোর জন্য মনে রাখতে হবে। আসুন দেখি কিভাবে আমরা Repository
পরিবর্তন করতে পারি আমাদের জন্য এই সমস্যার সমাধান করতে।
প্রধান নিরাপত্তার জন্য coroutines ব্যবহার করুন
আমরা একটি ফাংশনকে প্রধান-নিরাপদ বিবেচনা করি যখন এটি মূল থ্রেডে UI আপডেটগুলিকে ব্লক করে না। makeLoginRequest
ফাংশন প্রধান-নিরাপদ নয়, কারণ মেইন থ্রেড থেকে makeLoginRequest
কল করা UI ব্লক করে। একটি ভিন্ন থ্রেডে একটি coroutine কার্যকরী স্থানান্তর করতে coroutines লাইব্রেরি থেকে withContext()
ফাংশন ব্যবহার করুন:
class LoginRepository(...) {
...
suspend fun makeLoginRequest(
jsonBody: String
): Result<LoginResponse> {
// Move the execution of the coroutine to the I/O dispatcher
return withContext(Dispatchers.IO) {
// Blocking network request code
}
}
}
withContext(Dispatchers.IO)
একটি I/O থ্রেডে coroutine কার্যকর করে, আমাদের কলিং ফাংশনকে প্রধান-নিরাপদ করে এবং UI কে প্রয়োজন অনুযায়ী আপডেট করতে সক্ষম করে।
makeLoginRequest
এছাড়াও suspend
কীওয়ার্ড দিয়ে চিহ্নিত করা হয়েছে। এই কীওয়ার্ডটি হল কোটলিনের একটি কোরোটিনের মধ্যে থেকে কল করার জন্য একটি ফাংশন কার্যকর করার উপায়।
নিম্নলিখিত উদাহরণে, LoginViewModel
এ coroutine তৈরি করা হয়েছে। যেহেতু makeLoginRequest
এক্সিকিউশনকে মূল থ্রেড থেকে সরিয়ে দেয়, login
ফাংশনের coroutine এখন মূল থ্রেডে কার্যকর করা যেতে পারে:
class LoginViewModel(
private val loginRepository: LoginRepository
): ViewModel() {
fun login(username: String, token: String) {
// Create a new coroutine on the UI thread
viewModelScope.launch {
val jsonBody = "{ username: \"$username\", token: \"$token\"}"
// Make the network call and suspend execution until it finishes
val result = loginRepository.makeLoginRequest(jsonBody)
// Display result of the network request to the user
when (result) {
is Result.Success<LoginResponse> -> // Happy path
else -> // Show error in UI
}
}
}
}
মনে রাখবেন যে coroutine এখনও এখানে প্রয়োজন, যেহেতু makeLoginRequest
একটি suspend
ফাংশন, এবং সমস্ত suspend
ফাংশন অবশ্যই একটি coroutine-এ কার্যকর করা উচিত।
এই কোডটি আগের login
উদাহরণ থেকে কয়েকটি উপায়ে আলাদা:
-
launch
একটিDispatchers.IO
প্যারামিটার নেয় না। আপনি যখনlaunch
জন্য একটিDispatcher
পাস করেন না, তখনviewModelScope
থেকে লঞ্চ করা যেকোন কোরোটিন মূল থ্রেডে চলে। - নেটওয়ার্ক অনুরোধের ফলাফল এখন সাফল্য বা ব্যর্থতার UI প্রদর্শন করতে পরিচালনা করা হয়।
লগইন ফাংশন এখন নিম্নরূপ সঞ্চালিত হয়:
- অ্যাপটি মূল থ্রেডের
View
লেয়ার থেকেlogin()
ফাংশনটিকে কল করে। -
launch
মূল থ্রেডে একটি নতুন coroutine তৈরি করে, এবং coroutine কার্যকর করা শুরু করে। - coroutine-এর মধ্যে,
loginRepository.makeLoginRequest()
এ করা কল এখন coroutine-এর আরও এক্সিকিউশন স্থগিত করে যতক্ষণ নাmakeLoginRequest()
-এwithContext
ব্লক চালু না হয়। - একবার
withContext
ব্লক শেষ হয়ে গেলে,login()
-এ coroutine নেটওয়ার্ক অনুরোধের ফলাফলের সাথে মূল থ্রেডে পুনরায় এক্সিকিউশন শুরু করে।
ব্যতিক্রম হ্যান্ডলিং
Repository
স্তর যে ব্যতিক্রমগুলি নিক্ষেপ করতে পারে তা পরিচালনা করতে, ব্যতিক্রমগুলির জন্য Kotlin এর অন্তর্নির্মিত সমর্থন ব্যবহার করুন। নিম্নলিখিত উদাহরণে, আমরা একটি try-catch
ব্লক ব্যবহার করি:
class LoginViewModel(
private val loginRepository: LoginRepository
): ViewModel() {
fun login(username: String, token: String) {
viewModelScope.launch {
val jsonBody = "{ username: \"$username\", token: \"$token\"}"
val result = try {
loginRepository.makeLoginRequest(jsonBody)
} catch(e: Exception) {
Result.Error(Exception("Network request failed"))
}
when (result) {
is Result.Success<LoginResponse> -> // Happy path
else -> // Show error in UI
}
}
}
}
এই উদাহরণে, makeLoginRequest()
কল দ্বারা নিক্ষিপ্ত কোনো অপ্রত্যাশিত ব্যতিক্রম UI-তে একটি ত্রুটি হিসাবে পরিচালনা করা হয়।
অতিরিক্ত coroutines সম্পদ
অ্যান্ড্রয়েডে কোরোটিনগুলির আরও বিশদ বিবরণের জন্য, কোটলিন কোরোটিনগুলির সাথে অ্যাপের কার্যক্ষমতা উন্নত করুন দেখুন।
আরও কোরোটিন সংস্থানগুলির জন্য, নিম্নলিখিত লিঙ্কগুলি দেখুন:
- Coroutines ওভারভিউ (JetBrains)
- Coroutines গাইড (JetBrains)
- কোটলিন কোরোটিন এবং প্রবাহের জন্য অতিরিক্ত সংস্থান