Android ऐप्लिकेशन, Android सिस्टम और अन्य Android ऐप्लिकेशन से ब्रॉडकास्ट मैसेज भेजते और पाते हैं. यह पब्लिश-सदस्यता डिज़ाइन पैटर्न की तरह ही होता है. आम तौर पर, सिस्टम और ऐप्लिकेशन कुछ खास इवेंट होने पर ब्रॉडकास्ट भेजते हैं. उदाहरण के लिए, Android सिस्टम, सिस्टम के अलग-अलग इवेंट होने पर ब्रॉडकास्ट भेजता है. जैसे, सिस्टम बूट होने या डिवाइस चार्ज होने पर. ऐप्लिकेशन, कस्टम ब्रॉडकास्ट भी भेजते हैं. उदाहरण के लिए, अन्य ऐप्लिकेशन को किसी ऐसी जानकारी की सूचना देने के लिए जिससे उनकी दिलचस्पी हो सकती है. जैसे, नया डेटा डाउनलोड होना.
ऐप्लिकेशन, खास ब्रॉडकास्ट पाने के लिए रजिस्टर कर सकते हैं. ब्रॉडकास्ट भेजने पर, सिस्टम अपने-आप उन ऐप्लिकेशन पर ब्रॉडकास्ट भेजता है जिन्होंने उस तरह के ब्रॉडकास्ट पाने के लिए सदस्यता ली है.
आम तौर पर, ब्रॉडकास्ट का इस्तेमाल सभी ऐप्लिकेशन में मैसेजिंग सिस्टम के तौर पर किया जा सकता है. साथ ही, इसे सामान्य उपयोगकर्ता फ़्लो के बाहर भी इस्तेमाल किया जा सकता है. हालांकि, आपको इस बात का ध्यान रखना चाहिए कि ब्रॉडकास्ट के जवाब देने और बैकग्राउंड में ऐसी जॉब चलाने के अवसर का गलत इस्तेमाल न किया जाए जिनसे सिस्टम की परफ़ॉर्मेंस पर असर पड़ सकता है.
सिस्टम ब्रॉडकास्ट के बारे में जानकारी
सिस्टम में अलग-अलग इवेंट होने पर, वह अपने-आप ब्रॉडकास्ट भेजता है. जैसे, जब सिस्टम हवाई जहाज़ मोड में स्विच करता है और उससे बाहर निकलता है. सदस्यता वाले सभी ऐप्लिकेशन को ये ब्रॉडकास्ट मिलते हैं.
Intent
ऑब्जेक्ट, ब्रॉडकास्ट मैसेज को रैप करता है. action
स्ट्रिंग, android.intent.action.AIRPLANE_MODE
जैसे हुए इवेंट की पहचान करती है. इंटेंट में, उसके अतिरिक्त फ़ील्ड में बंडल की गई अतिरिक्त जानकारी भी शामिल हो सकती है.
उदाहरण के लिए, हवाई जहाज़ मोड के इंटेंट में एक बूलियन एक्सट्रा शामिल होता है, जो यह बताता है कि हवाई जहाज़ मोड चालू है या नहीं.
इंटेंट पढ़ने और किसी इंटेंट से कार्रवाई की स्ट्रिंग पाने के तरीके के बारे में ज़्यादा जानने के लिए, इंटेंट और इंटेंट फ़िल्टर लेख पढ़ें.
सिस्टम ब्रॉडकास्ट से जुड़ी कार्रवाइयां
सिस्टम ब्रॉडकास्ट ऐक्शन की पूरी सूची के लिए, Android SDK में BROADCAST_ACTIONS.TXT
फ़ाइल देखें. हर ब्रॉडकास्ट ऐक्शन के साथ एक कॉन्स्टेंट फ़ील्ड जुड़ा होता है. उदाहरण के लिए, स्थिर वैल्यू ACTION_AIRPLANE_MODE_CHANGED
की वैल्यू android.intent.action.AIRPLANE_MODE
है.
हर ब्रॉडकास्ट ऐक्शन का दस्तावेज़, उससे जुड़े कॉन्स्टेंट फ़ील्ड में उपलब्ध होता है.
सिस्टम ब्रॉडकास्ट में बदलाव
Android प्लैटफ़ॉर्म के साथ-साथ, सिस्टम ब्रॉडकास्ट के काम करने का तरीका भी समय-समय पर बदलता रहता है. Android के सभी वर्शन के साथ काम करने के लिए, इन बदलावों का ध्यान रखें.
Android 14
जब ऐप्लिकेशन कैश मेमोरी में सेव होने की स्थिति में होते हैं, तब सिस्टम की परफ़ॉर्मेंस को बेहतर बनाने के लिए, सिस्टम ब्रॉडकास्ट डिलीवरी को ऑप्टिमाइज़ करता है. उदाहरण के लिए, ऐप्लिकेशन के कैश मेमोरी में मौजूद होने पर, सिस्टम ACTION_SCREEN_ON
जैसे कम अहम सिस्टम ब्रॉडकास्ट को बाद में दिखाता है.
जब ऐप्लिकेशन कैश मेमोरी में सेव होने की स्थिति से ऐक्टिव प्रोसेस लाइफ़साइकल में चला जाता है, तब सिस्टम, देर से भेजे गए ब्रॉडकास्ट डिलीवर करता है.
मेनिफ़ेस्ट में बताए गए अहम ब्रॉडकास्ट, डिलीवरी के लिए ऐप्लिकेशन को कुछ समय के लिए कैश मेमोरी से हटा देते हैं.
Android 9
Android 9 (एपीआई लेवल 28) से, NETWORK_STATE_CHANGED_ACTION
ब्रॉडकास्ट को उपयोगकर्ता की जगह या व्यक्तिगत पहचान से जुड़ी जानकारी नहीं मिलती.
अगर आपका ऐप्लिकेशन, Android 9.0 (एपीआई लेवल 28) या इससे बाद के वर्शन पर चलने वाले डिवाइस पर इंस्टॉल है, तो सिस्टम में वाई-फ़ाई ब्रॉडकास्ट में एसएसआईडी, बीएसएसआईडी, कनेक्शन की जानकारी या स्कैन के नतीजे शामिल नहीं होते. यह जानकारी पाने के लिए, getConnectionInfo()
को कॉल करें.
Android 8.0
Android 8.0 (एपीआई लेवल 26) से, सिस्टम मेनिफ़ेस्ट में बताए गए रिसीवर पर अतिरिक्त पाबंदियां लगाता है.
अगर आपका ऐप्लिकेशन Android 8.0 या इसके बाद के वर्शन को टारगेट करता है, तो मेनिफ़ेस्ट का इस्तेमाल करके, ज़्यादातर इंप्लिसिट ब्रॉडकास्ट (ऐसे ब्रॉडकास्ट जो खास तौर पर आपके ऐप्लिकेशन को टारगेट नहीं करते) के लिए रिसीवर का एलान नहीं किया जा सकता. जब कोई उपयोगकर्ता आपके ऐप्लिकेशन का इस्तेमाल कर रहा हो, तब भी कॉन्टेक्स्ट के हिसाब से रजिस्टर किए गए रिसीवर का इस्तेमाल किया जा सकता है.
Android 7.0
Android 7.0 (एपीआई लेवल 24) और इसके बाद के वर्शन, ये सिस्टम ब्रॉडकास्ट नहीं भेजते:
साथ ही, Android 7.0 और इसके बाद के वर्शन को टारगेट करने वाले ऐप्लिकेशन को, registerReceiver(BroadcastReceiver, IntentFilter)
का इस्तेमाल करके CONNECTIVITY_ACTION
ब्रॉडकास्ट को रजिस्टर करना होगा. मेनिफ़ेस्ट में रिसीवर का एलान करने से काम नहीं चलता.
ब्रॉडकास्ट पाना
ऐप्लिकेशन को ब्रॉडकास्ट दो तरीकों से मिल सकते हैं: कॉन्टेक्स्ट के हिसाब से रजिस्टर किए गए रिसीवर और मेनिफ़ेस्ट में बताए गए रिसीवर के ज़रिए.
कॉन्टेक्स्ट के हिसाब से रजिस्टर किए गए ईमेल पाने वाले लोग
संदर्भ के हिसाब से रजिस्टर किए गए रिसीवर को तब तक ब्रॉडकास्ट मिलते हैं, जब तक उनका रजिस्टर किया गया संदर्भ मान्य होता है. आम तौर पर, यह registerReceiver
और
unregisterReceiver
के बीच होता है. जब सिस्टम उस कॉन्टेक्स्ट को मिटा देता है जिसे रजिस्टर किया जा रहा है, तो रजिस्टर किया जा रहा कॉन्टेक्स्ट भी अमान्य हो जाता है. उदाहरण के लिए, अगर आपने Activity
कॉन्टेक्स्ट में रजिस्टर किया है, तो गतिविधि के चालू रहने तक आपको ब्रॉडकास्ट मिलते रहेंगे. ऐप्लिकेशन कॉन्टेक्स्ट के साथ रजिस्टर करने पर, ऐप्लिकेशन के चलने तक आपको ब्रॉडकास्ट मिलते रहेंगे.
किसी रिसीवर को संदर्भ के साथ रजिस्टर करने के लिए, यह तरीका अपनाएं:
अपने ऐप्लिकेशन के मॉड्यूल-लेवल की बिल्ड फ़ाइल में, AndroidX कोर लाइब्रेरी का 1.9.0 या उसके बाद का वर्शन शामिल करें:
ग्रूवी
dependencies { def core_version = "1.13.1" // Java language implementation implementation "androidx.core:core:$core_version" // Kotlin implementation "androidx.core:core-ktx:$core_version" // To use RoleManagerCompat implementation "androidx.core:core-role:1.0.0" // To use the Animator APIs implementation "androidx.core:core-animation:1.0.0" // To test the Animator APIs androidTestImplementation "androidx.core:core-animation-testing:1.0.0" // Optional - To enable APIs that query the performance characteristics of GMS devices. implementation "androidx.core:core-performance:1.0.0" // Optional - to use ShortcutManagerCompat to donate shortcuts to be used by Google implementation "androidx.core:core-google-shortcuts:1.1.0" // Optional - to support backwards compatibility of RemoteViews implementation "androidx.core:core-remoteviews:1.1.0" // Optional - APIs for SplashScreen, including compatibility helpers on devices prior Android 12 implementation "androidx.core:core-splashscreen:1.2.0-alpha02" }
Kotlin
dependencies { val core_version = "1.13.1" // Java language implementation implementation("androidx.core:core:$core_version") // Kotlin implementation("androidx.core:core-ktx:$core_version") // To use RoleManagerCompat implementation("androidx.core:core-role:1.0.0") // To use the Animator APIs implementation("androidx.core:core-animation:1.0.0") // To test the Animator APIs androidTestImplementation("androidx.core:core-animation-testing:1.0.0") // Optional - To enable APIs that query the performance characteristics of GMS devices. implementation("androidx.core:core-performance:1.0.0") // Optional - to use ShortcutManagerCompat to donate shortcuts to be used by Google implementation("androidx.core:core-google-shortcuts:1.1.0") // Optional - to support backwards compatibility of RemoteViews implementation("androidx.core:core-remoteviews:1.1.0") // Optional - APIs for SplashScreen, including compatibility helpers on devices prior Android 12 implementation("androidx.core:core-splashscreen:1.2.0-alpha02") }
BroadcastReceiver
का इंस्टेंस बनाएं:Kotlin
val myBroadcastReceiver = MyBroadcastReceiver()
Java
MyBroadcastReceiver myBroadcastReceiver = new MyBroadcastReceiver();
IntentFilter
का इंस्टेंस बनाएं:Kotlin
val filter = IntentFilter("com.example.snippets.ACTION_UPDATE_DATA")
Java
IntentFilter filter = new IntentFilter("com.example.snippets.ACTION_UPDATE_DATA");
चुनें कि ब्रॉडकास्ट रिसीवर को एक्सपोर्ट किया जाना चाहिए या नहीं. साथ ही, यह भी चुनें कि डिवाइस पर मौजूद अन्य ऐप्लिकेशन को यह दिखना चाहिए या नहीं. अगर यह रिसीवर, सिस्टम या अन्य ऐप्लिकेशन से भेजे गए ब्रॉडकास्ट सुन रहा है, तो
RECEIVER_EXPORTED
फ़्लैग का इस्तेमाल करें. भले ही, वे ऐप्लिकेशन आपके मालिकाना हक वाले हों. अगर यह रिसीवर सिर्फ़ आपके ऐप्लिकेशन से भेजे गए ब्रॉडकास्ट को सुन रहा है, तोRECEIVER_NOT_EXPORTED
फ़्लैग का इस्तेमाल करें.Kotlin
val listenToBroadcastsFromOtherApps = false val receiverFlags = if (listenToBroadcastsFromOtherApps) { ContextCompat.RECEIVER_EXPORTED } else { ContextCompat.RECEIVER_NOT_EXPORTED }
Java
boolean listenToBroadcastsFromOtherApps = false; int receiverFlags = listenToBroadcastsFromOtherApps ? ContextCompat.RECEIVER_EXPORTED : ContextCompat.RECEIVER_NOT_EXPORTED;
registerReceiver()
को कॉल करके, पैसे पाने वाले व्यक्ति को रजिस्टर करें:Kotlin
ContextCompat.registerReceiver(context, myBroadcastReceiver, filter, receiverFlags)
Java
ContextCompat.registerReceiver(context, myBroadcastReceiver, filter, receiverFlags);
ब्रॉडकास्ट पाने की सुविधा बंद करने के लिए,
unregisterReceiver(android.content.BroadcastReceiver)
पर कॉल करें. जब आपको रिसीवर की ज़रूरत न पड़े या संदर्भ अब मान्य न हो, तो उसे रजिस्टर करना न भूलें.
अपने ब्रॉडकास्ट रिसीवर को अनरजिस्टर करना
ब्रॉडकास्ट रिसीवर रजिस्टर होने के दौरान, वह उस कॉन्टेक्स्ट का रेफ़रंस रखता है जिसके साथ आपने उसे रजिस्टर किया है. अगर पाने वाले के रजिस्टर किए गए स्कोप की अवधि, संदर्भ के लाइफ़साइकल के स्कोप से ज़्यादा है, तो डेटा लीक हो सकता है. उदाहरण के लिए, ऐसा तब हो सकता है, जब आपने किसी गतिविधि के दायरे में रिसीवर को रजिस्टर किया हो, लेकिन सिस्टम गतिविधि को मिटाने पर, आपने उसे अनरजिस्टर करना भूल जाएं. इसलिए, अपने ब्रॉडकास्ट रिसीवर को हमेशा रजिस्टर करें.
Kotlin
class MyActivity : ComponentActivity() {
private val myBroadcastReceiver = MyBroadcastReceiver()
override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
// ...
ContextCompat.registerReceiver(this, myBroadcastReceiver, filter, receiverFlags)
setContent { MyApp() }
}
override fun onDestroy() {
super.onDestroy()
// When you forget to unregister your receiver here, you're causing a leak!
this.unregisterReceiver(myBroadcastReceiver)
}
}
Java
class MyActivity extends ComponentActivity {
MyBroadcastReceiver myBroadcastReceiver;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
// ...
ContextCompat.registerReceiver(this, myBroadcastReceiver, filter, receiverFlags);
// Set content
}
}
रिसीवर को सबसे छोटे दायरे में रजिस्टर करना
आपका ब्रॉडकास्ट रिसीवर सिर्फ़ तब रजिस्टर किया जाना चाहिए, जब आपको असल में नतीजे में दिलचस्पी हो. रिसीवर का सबसे छोटा स्कोप चुनें:
LifecycleResumeEffect
या गतिविधिonResume
/onPause
के लाइफ़साइकल के तरीकों: ब्रॉडकास्ट रिसीवर को सिर्फ़ तब अपडेट मिलते हैं, जब ऐप्लिकेशन फिर से शुरू होने की स्थिति में हो.LifecycleStartEffect
या गतिविधिonStart
/onStop
के लाइफ़साइकल के तरीकों: ब्रॉडकास्ट रिसीवर को सिर्फ़ तब अपडेट मिलते हैं, जब ऐप्लिकेशन फिर से शुरू होने की स्थिति में हो.DisposableEffect
: ब्रॉडकास्ट रिसीवर को अपडेट सिर्फ़ तब मिलते हैं, जब कॉम्पोज़ेबल, कॉम्पोज़िशन ट्री में होता है. यह स्कोप, ऐक्टिविटी के लाइफ़साइकल के स्कोप से अटैच नहीं है. रिसीवर को ऐप्लिकेशन कॉन्टेक्स्ट पर रजिस्टर करें. ऐसा इसलिए है, क्योंकि कॉम्पोज़ेबल, सैद्धांतिक तौर पर ऐक्टिविटी लाइफ़साइकल के दायरे से बाहर हो सकता है और ऐक्टिविटी को लीक कर सकता है.- गतिविधि
onCreate
/onDestroy
: गतिविधि बनने के बाद, ब्रॉडकास्ट रिसीवर को अपडेट मिलते हैं.onDestroy()
में रजिस्टर करें, न किonSaveInstanceState(Bundle)
में, क्योंकि हो सकता है किonSaveInstanceState(Bundle)
को कॉल न किया जाए. - कस्टम स्कोप: उदाहरण के लिए, अपने
ViewModel
स्कोप में रिसीवर को रजिस्टर किया जा सकता है, ताकि गतिविधि फिर से शुरू होने पर भी वह काम करता रहे. रिसीवर को रजिस्टर करने के लिए, ऐप्लिकेशन के कॉन्टेक्स्ट का इस्तेमाल करना न भूलें. ऐसा इसलिए, क्योंकि रिसीवर, गतिविधि के लाइफ़साइकल के दायरे से बाहर जा सकता है और गतिविधि को लीक कर सकता है.
स्टेटफ़ुल और स्टेटलेस कॉम्पोज़ेबल बनाना
Compose में स्टेटफ़ुल और स्टेटलेस कॉम्पोज़ेबल होते हैं. किसी कॉम्पोज़ेबल में ब्रॉडकास्ट रिसीवर को रजिस्टर या अनरजिस्टर करने से, वह स्टेटफ़ुल हो जाता है. कंपोज़ेबल, ऐसा फ़ंक्शन नहीं है जो एक जैसे पैरामीटर इस्तेमाल करने पर एक जैसा कॉन्टेंट रेंडर करता है. रजिस्टर किए गए ब्रॉडकास्ट रिसीवर को किए गए कॉल के आधार पर, इंटरनल स्टेटस बदल सकता है.
हमारा सुझाव है कि Compose में, अपने कॉम्पोज़ेबल को स्टेटफ़ुल और स्टैटलेस वर्शन में बांटें. इसलिए, हमारा सुझाव है कि आप ब्रॉडकास्ट रिसीवर को Composable से बाहर बनाएं, ताकि इसे स्टेटलेस बनाया जा सके:
@Composable
fun MyStatefulScreen() {
val myBroadcastReceiver = remember { MyBroadcastReceiver() }
val context = LocalContext.current
LifecycleStartEffect(true) {
// ...
ContextCompat.registerReceiver(context, myBroadcastReceiver, filter, flags)
onStopOrDispose { context.unregisterReceiver(myBroadcastReceiver) }
}
MyStatelessScreen()
}
@Composable
fun MyStatelessScreen() {
// Implement your screen
}
मेनिफ़ेस्ट में बताए गए रिसीवर
अगर आपने अपने मेनिफ़ेस्ट में ब्रॉडकास्ट रिसीवर का एलान किया है, तो ब्रॉडकास्ट भेजे जाने पर सिस्टम आपका ऐप्लिकेशन लॉन्च करेगा. अगर ऐप्लिकेशन पहले से नहीं चल रहा है, तो सिस्टम ऐप्लिकेशन को लॉन्च करता है.
मेनिफ़ेस्ट में ब्रॉडकास्ट रिसीवर का एलान करने के लिए, यह तरीका अपनाएं:
अपने ऐप्लिकेशन के मेनिफ़ेस्ट में
<receiver>
एलिमेंट की जानकारी दें.<!-- If this receiver listens for broadcasts sent from the system or from other apps, even other apps that you own, set android:exported to "true". --> <receiver android:name=".MyBroadcastReceiver" android:exported="false"> <intent-filter> <action android:name="com.example.snippets.ACTION_UPDATE_DATA" /> </intent-filter> </receiver>
इंटेंट फ़िल्टर, उन ब्रॉडकास्ट ऐक्शन के बारे में बताते हैं जिनकी सदस्यता आपका रिसीवर लेता है.
सब-क्लास
BroadcastReceiver
बनाएं औरonReceive(Context, Intent)
लागू करें. यहां दिए गए उदाहरण में, ब्रॉडकास्ट रिसीवर, ब्रॉडकास्ट के कॉन्टेंट को लॉग करता है और दिखाता है:Kotlin
class MyBroadcastReceiver : BroadcastReceiver() { @Inject lateinit var dataRepository: DataRepository override fun onReceive(context: Context, intent: Intent) { if (intent.action == "com.example.snippets.ACTION_UPDATE_DATA") { val data = intent.getStringExtra("com.example.snippets.DATA") ?: "No data" // Do something with the data, for example send it to a data repository: dataRepository.updateData(data) } } }
Java
public static class MyBroadcastReceiver extends BroadcastReceiver { @Inject DataRepository dataRepository; @Override public void onReceive(Context context, Intent intent) { if (Objects.equals(intent.getAction(), "com.example.snippets.ACTION_UPDATE_DATA")) { String data = intent.getStringExtra("com.example.snippets.DATA"); // Do something with the data, for example send it to a data repository: if (data != null) { dataRepository.updateData(data); } } } }
ऐप्लिकेशन इंस्टॉल होने पर, सिस्टम पैकेज मैनेजर रिसीवर को रजिस्टर करता है. इसके बाद, रिसीवर आपके ऐप्लिकेशन का एक अलग एंट्री पॉइंट बन जाता है. इसका मतलब है कि अगर ऐप्लिकेशन नहीं चल रहा है, तो सिस्टम ऐप्लिकेशन को शुरू करके ब्रॉडकास्ट डिलीवर कर सकता है.
सिस्टम, मिलने वाले हर ब्रॉडकास्ट को मैनेज करने के लिए, एक नया BroadcastReceiver
कॉम्पोनेंट ऑब्जेक्ट बनाता है. यह ऑब्जेक्ट सिर्फ़ onReceive(Context, Intent)
को कॉल करने के दौरान मान्य होता है. इस तरीके से आपका कोड वापस आने के बाद, सिस्टम उस कॉम्पोनेंट को अब चालू नहीं मानता.
प्रोसेस की स्थिति पर असर
BroadcastReceiver
के चालू होने या न होने से, उसमें मौजूद प्रोसेस पर असर पड़ता है. इससे, सिस्टम के क्रैश होने की संभावना में बदलाव हो सकता है. फ़ोरग्राउंड प्रोसेस, पैसे पाने वाले व्यक्ति के onReceive()
मेथड को लागू करती है. सिस्टम, मेमोरी के ज़्यादा लोड होने पर छोड़कर, प्रोसेस को चलाता है.
सिस्टम, onReceive()
के बाद BroadcastReceiver
को बंद कर देता है.
रिसीवर की होस्ट प्रोसेस का महत्व, उसके ऐप्लिकेशन कॉम्पोनेंट पर निर्भर करता है. अगर उस प्रोसेस में सिर्फ़ मेनिफ़ेस्ट में बताए गए रिसीवर को होस्ट किया जाता है, तो सिस्टम onReceive()
के बाद उसे बंद कर सकता है, ताकि ज़्यादा ज़रूरी प्रोसेस के लिए संसाधनों को खाली किया जा सके. ऐसा उन ऐप्लिकेशन के लिए आम तौर पर होता है जिनका इस्तेमाल उपयोगकर्ता ने कभी नहीं किया है या हाल ही में नहीं किया है.
इसलिए, ब्रॉडकास्ट रिसीवर को लंबे समय तक चलने वाली बैकग्राउंड थ्रेड शुरू नहीं करनी चाहिए.
onReceive()
के बाद, सिस्टम किसी भी समय प्रोसेस को रोक सकता है, ताकि वह बनाई गई थ्रेड को बंद करके, मेमोरी वापस पा सके. प्रोसेस को जारी रखने के लिए, JobScheduler
का इस्तेमाल करके, रिसीवर से JobService
शेड्यूल करें, ताकि सिस्टम को पता रहे कि प्रोसेस अब भी जारी है. बैकग्राउंड में चल रहे काम की खास जानकारी में ज़्यादा जानकारी दी गई है.
ब्रॉडकास्ट भेजना
Android, ऐप्लिकेशन के लिए ब्रॉडकास्ट भेजने के दो तरीके उपलब्ध कराता है:
sendOrderedBroadcast(Intent, String)
वाला तरीका, एक बार में एक ही रिसीवर को ब्रॉडकास्ट भेजता है. हर रिसीवर के क्रम से लागू होने पर, वह अगले रिसीवर को नतीजा भेज सकता है. यह ब्रॉडकास्ट को पूरी तरह से बंद भी कर सकता है, ताकि वह दूसरे रिसीवर तक न पहुंच पाए. आपके पास यह कंट्रोल करने का विकल्प होता है कि रिसीवर किस क्रम में चलें. ऐसा करने के लिए, मैच करने वाले इंटेंट-फ़िल्टर के android:priority एट्रिब्यूट का इस्तेमाल करें. एक ही प्राथमिकता वाले रिसीवर, किसी भी क्रम में चलाए जाते हैं.sendBroadcast(Intent)
तरीका, ब्रॉडकास्ट को सभी पाने वालों को तय किए गए क्रम में नहीं भेजता. इसे सामान्य ब्रॉडकास्ट कहा जाता है. यह तरीका ज़्यादा असरदार है. हालांकि, इसका मतलब है कि रिसीवर, दूसरे रिसीवर के नतीजे नहीं पढ़ सकते, ब्रॉडकास्ट से मिले डेटा को प्रोपेगेट नहीं कर सकते या ब्रॉडकास्ट को बंद नहीं कर सकते.
नीचे दिया गया कोड स्निपेट, इंटेंट बनाकर और sendBroadcast(Intent)
को कॉल करके ब्रॉडकास्ट भेजने का तरीका दिखाता है.
Kotlin
val intent = Intent("com.example.snippets.ACTION_UPDATE_DATA").apply {
putExtra("com.example.snippets.DATA", newData)
setPackage("com.example.snippets")
}
context.sendBroadcast(intent)
Java
Intent intent = new Intent("com.example.snippets.ACTION_UPDATE_DATA");
intent.putExtra("com.example.snippets.DATA", newData);
intent.setPackage("com.example.snippets");
context.sendBroadcast(intent);
ब्रॉडकास्ट मैसेज को Intent
ऑब्जेक्ट में लपेटा जाता है. इंटेंट की action
स्ट्रिंग में, ऐप्लिकेशन के Java पैकेज के नाम का सिंटैक्स होना चाहिए. साथ ही, ब्रॉडकास्ट इवेंट की यूनीक पहचान होनी चाहिए. putExtra(String, Bundle)
का इस्तेमाल करके, इंटेंट में ज़्यादा जानकारी अटैच की जा सकती है. इंटेंट पर setPackage(String)
को कॉल करके, किसी ब्रॉडकास्ट को एक ही संगठन के ऐप्लिकेशन के सेट तक सीमित किया जा सकता है.
अनुमतियों की मदद से ब्रॉडकास्ट पर पाबंदी लगाना
अनुमतियों की मदद से, ब्रॉडकास्ट को उन ऐप्लिकेशन के सेट तक सीमित किया जा सकता है जिनके पास कुछ अनुमतियां हैं. ब्रॉडकास्ट भेजने वाले या पाने वाले, दोनों पर पाबंदियां लगाई जा सकती हैं.
अनुमतियों के साथ ब्रॉडकास्ट भेजना
sendBroadcast(Intent, String)
या
sendOrderedBroadcast(Intent, String, BroadcastReceiver, Handler, int, String,
Bundle)
को कॉल करते समय, अनुमति पैरामीटर की जानकारी दी जा सकती है. ब्रॉडकास्ट सिर्फ़ उन रिसीवर को मिल सकता है जिन्होंने अपने मेनिफ़ेस्ट में <uses-permission>
टैग के साथ अनुमति का अनुरोध किया है. अगर अनुमति खतरनाक है, तो ब्रॉडकास्ट पाने से पहले, आपको अनुमति देनी होगी. उदाहरण के लिए, यह कोड अनुमति के साथ ब्रॉडकास्ट भेजता है:
Kotlin
context.sendBroadcast(intent, android.Manifest.permission.ACCESS_COARSE_LOCATION)
Java
context.sendBroadcast(intent, android.Manifest.permission.ACCESS_COARSE_LOCATION);
ब्रॉडकास्ट पाने के लिए, उसे अनुमति का अनुरोध इस तरह करना होगा:
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
आपके पास BLUETOOTH_CONNECT
जैसी किसी मौजूदा सिस्टम की अनुमति तय करने या <permission>
एलिमेंट की मदद से कस्टम अनुमति तय करने का विकल्प होता है. अनुमतियों और सुरक्षा के बारे में सामान्य जानकारी के लिए, सिस्टम की अनुमतियां देखें.
अनुमतियों के साथ ब्रॉडकास्ट पाना
अगर ब्रॉडकास्ट रिसीवर को रजिस्टर करते समय, अनुमति पैरामीटर (अपने मेनिफ़ेस्ट में registerReceiver(BroadcastReceiver, IntentFilter, String, Handler)
या <receiver>
टैग के साथ) दिया जाता है, तो सिर्फ़ वे ब्रॉडकास्टर रिसीवर को इंटेंट भेज सकते हैं जिन्होंने अपने मेनिफ़ेस्ट में <uses-permission>
टैग के साथ अनुमति का अनुरोध किया है. अगर अनुमति खतरनाक है, तो ब्रॉडकास्टर को भी अनुमति देनी होगी.
उदाहरण के लिए, मान लें कि आपके ऐप्लिकेशन में मेनिफ़ेस्ट में एलान किया गया रिसीवर इस तरह है:
<!-- If this receiver listens for broadcasts sent from the system or from
other apps, even other apps that you own, set android:exported to "true". -->
<receiver
android:name=".MyBroadcastReceiverWithPermission"
android:permission="android.permission.ACCESS_COARSE_LOCATION"
android:exported="true">
<intent-filter>
<action android:name="com.example.snippets.ACTION_UPDATE_DATA" />
</intent-filter>
</receiver>
इसके अलावा, आपके ऐप्लिकेशन में कॉन्टेक्स्ट के हिसाब से रजिस्टर किया गया रिसीवर इस तरह से हो:
Kotlin
ContextCompat.registerReceiver(
context, myBroadcastReceiver, filter,
android.Manifest.permission.ACCESS_COARSE_LOCATION,
null, // scheduler that defines thread, null means run on main thread
receiverFlags
)
Java
ContextCompat.registerReceiver(
context, myBroadcastReceiver, filter,
android.Manifest.permission.ACCESS_COARSE_LOCATION,
null, // scheduler that defines thread, null means run on main thread
receiverFlags
);
इसके बाद, उन रिसीवर को ब्रॉडकास्ट भेजने के लिए, भेजने वाले ऐप्लिकेशन को अनुमति का अनुरोध इस तरह करना होगा:
<uses-permission android:name="android.permission.ACCESS_COARSE_LOCATION" />
सुरक्षा से जुड़ी बातें
ब्रॉडकास्ट भेजने और पाने के लिए, सुरक्षा से जुड़ी कुछ बातें ध्यान में रखें:
अगर कई ऐप्लिकेशन ने अपने मेनिफ़ेस्ट में एक ही ब्रॉडकास्ट पाने के लिए रजिस्टर किया है, तो हो सकता है कि सिस्टम कई ऐप्लिकेशन लॉन्च कर दे. इससे डिवाइस की परफ़ॉर्मेंस और उपयोगकर्ता अनुभव, दोनों पर काफ़ी असर पड़ सकता है. इससे बचने के लिए, मेनिफ़ेस्ट एलान के बजाय कॉन्टेक्स्ट रजिस्ट्रेशन का इस्तेमाल करें. कभी-कभी, Android सिस्टम खुद ही कॉन्टेक्स्ट के हिसाब से रजिस्टर किए गए रिसीवर का इस्तेमाल करता है. उदाहरण के लिए,
CONNECTIVITY_ACTION
ब्रॉडकास्ट सिर्फ़ उन रिसीवर को डिलीवर किया जाता है जो कॉन्टेक्स्ट के हिसाब से रजिस्टर किए गए हैं.इंप्लिसिट इंटेंट का इस्तेमाल करके, संवेदनशील जानकारी ब्रॉडकास्ट न करें. अगर कोई ऐप्लिकेशन ब्रॉडकास्ट पाने के लिए रजिस्टर करता है, तो वह जानकारी पढ़ सकता है. यह कंट्रोल करने के तीन तरीके हैं कि आपके ब्रॉडकास्ट किसे मिलें:
- ब्रॉडकास्ट भेजते समय, अनुमति तय की जा सकती है.
- Android 4.0 (एपीआई लेवल 14) और उसके बाद के वर्शन में, ब्रॉडकास्ट भेजते समय
setPackage(String)
के साथ पैकेज तय किया जा सकता है. सिस्टम, ब्रॉडकास्ट को पैकेज से मैच करने वाले ऐप्लिकेशन के सेट तक सीमित कर देता है.
रिसीवर को रजिस्टर करने पर, कोई भी ऐप्लिकेशन आपके ऐप्लिकेशन के रिसीवर पर नुकसान पहुंचाने वाले ब्रॉडकास्ट भेज सकता है. आपके ऐप्लिकेशन को मिलने वाले ब्रॉडकास्ट को सीमित करने के कई तरीके हैं:
- ब्रॉडकास्ट रिसीवर को रजिस्टर करते समय, अनुमति तय की जा सकती है.
- मेनिफ़ेस्ट में बताए गए रिसीवर के लिए, मेनिफ़ेस्ट में android:exported एट्रिब्यूट को "गलत" पर सेट किया जा सकता है. रिसीवर को ऐप्लिकेशन के बाहर के सोर्स से ब्रॉडकास्ट नहीं मिलते.
ब्रॉडकास्ट ऐक्शन के लिए नेमस्पेस ग्लोबल होता है. पक्का करें कि कार्रवाई के नाम और अन्य स्ट्रिंग, आपके मालिकाना हक वाले नेमस्पेस में लिखी गई हों. ऐसा न करने पर, अनजाने में दूसरे ऐप्लिकेशन के साथ गड़बड़ी हो सकती है.
रिसीवर का
onReceive(Context, Intent)
तरीका मुख्य थ्रेड पर चलता है, इसलिए इसे तेज़ी से लागू किया जाना चाहिए और तेज़ी से रिटर्न किया जाना चाहिए. अगर आपको लंबे समय तक चलने वाला काम करना है, तो थ्रेड बनाने या बैकग्राउंड सेवाएं शुरू करने में सावधानी बरतें. ऐसा इसलिए, क्योंकिonReceive()
के वापस आने के बाद, सिस्टम पूरी प्रोसेस को बंद कर सकता है. ज़्यादा जानकारी के लिए, प्रोसेस की स्थिति पर असर देखें. लंबे समय तक चलने वाले काम करने के लिए, हमारा सुझाव है कि:- पाने वाले के
onReceive()
तरीके मेंgoAsync()
को कॉल करना औरBroadcastReceiver.PendingResult
को बैकग्राउंड थ्रेड में पास करना. इससेonReceive()
से वापस आने के बाद भी ब्रॉडकास्ट चालू रहता है. हालांकि, इस तरीके का इस्तेमाल करने पर भी, सिस्टम को उम्मीद है कि आपका ब्रॉडकास्ट बहुत जल्दी (10 सेकंड से कम समय में) खत्म हो जाएगा. इसकी मदद से, मुख्य थ्रेड में गड़बड़ी से बचने के लिए, काम को किसी दूसरी थ्रेड में ले जाया जा सकता है. JobScheduler
की मदद से कोई जॉब शेड्यूल करना. ज़्यादा जानकारी के लिए, स्मार्ट तरीके से जॉब शेड्यूल करना देखें.
- पाने वाले के
ब्रॉडकास्ट रिसीवर से गतिविधियां शुरू न करें, क्योंकि इससे उपयोगकर्ता अनुभव खराब होता है. खास तौर पर, अगर एक से ज़्यादा रिसीवर हैं. इसके बजाय, सूचना दिखाएं.