जवाबों:
मुझे यकीन नहीं है कि यदि आप इस धागे पर विचार करते हैं तो एपीआई सीधे एपीआई प्रदान करता है :
मैं उसी चीज़ पर हैरान हो रहा था।
मेरे मामले में मेरे पास एकBroadcastReceiver
कार्यान्वयन हैContext#unregisterReceiver(BroadcastReceiver)
जो इसे प्राप्त करने वाले इरादे को संभालने के बाद तर्क के रूप में स्वयं को पारित करता है।
एक छोटा सा मौका है कि रिसीवर कीonReceive(Context, Intent)
विधि को एक से अधिक बार कहा जाता है, क्योंकि यह कई के साथ पंजीकृत हैIntentFilters
, जिससेIllegalArgumentException
फेंके जाने की संभावना पैदा होती हैContext#unregisterReceiver(BroadcastReceiver)
।मेरे मामले में, मैं कॉल करने से पहले जांच करने के लिए एक निजी सिंक्रनाइज़ सदस्य को स्टोर कर सकता हूं
Context#unregisterReceiver(BroadcastReceiver)
, लेकिन अगर एपीआई एक चेक विधि प्रदान करता है तो यह बहुत साफ होगा।
यदि कोई रिसीवर पंजीकृत है, तो यह जांचने के लिए कोई एपीआई फ़ंक्शन नहीं है। वर्कअराउंड में अपना कोड डालना हैtry catch block as done below.
try {
//Register or UnRegister your broadcast receiver here
} catch(IllegalArgumentException e) {
e.printStackTrace();
}
null
। लेकिन जैसा कि आपने बताया, मैं जा रहा हूं try catch
। हास्यास्पद।
public class MyReceiver extends BroadcastReceiver {
public boolean isRegistered;
/**
* register receiver
* @param context - Context
* @param filter - Intent Filter
* @return see Context.registerReceiver(BroadcastReceiver,IntentFilter)
*/
public Intent register(Context context, IntentFilter filter) {
try {
// ceph3us note:
// here I propose to create
// a isRegistered(Contex) method
// as you can register receiver on different context
// so you need to match against the same one :)
// example by storing a list of weak references
// see LoadedApk.class - receiver dispatcher
// its and ArrayMap there for example
return !isRegistered
? context.registerReceiver(this, filter)
: null;
} finally {
isRegistered = true;
}
}
/**
* unregister received
* @param context - context
* @return true if was registered else false
*/
public boolean unregister(Context context) {
// additional work match on context before unregister
// eg store weak ref in register then compare in unregister
// if match same instance
return isRegistered
&& unregisterInternal(context);
}
private boolean unregisterInternal(Context context) {
context.unregisterReceiver(this);
isRegistered = false;
return true;
}
// rest implementation here
// or make this an abstract class as template :)
...
}
MyReceiver myReceiver = new MyReceiver();
myReceiver.register(Context, IntentFilter); // register
myReceiver.unregister(Context); // unregister
विज्ञापन १
-- प्रतिउत्तर में:
यह वास्तव में इतना सुरुचिपूर्ण नहीं है क्योंकि आपको पंजीकरण के बाद isRegistered ध्वज सेट करना याद रखना होगा। - चुपके रब्बी
- झंडे को पंजीकृत करने और सेट करने के लिए रिसीवर में एक और तरीका "जोड़ा गया तरीका"
यह काम नहीं करेगा यदि आप डिवाइस को पुनरारंभ करते हैं या यदि आपका ऐप ओएस द्वारा मारा गया है। - 6 घंटे पहले
@amin - कोड में जीवनकाल देखें (न कि सिस्टम, जो कि उपस्थिति प्रविष्टि द्वारा पंजीकृत है) पंजीकृत रिसीवर :)
मैं इस समाधान का उपयोग कर रहा हूं
public class ReceiverManager {
private static List<BroadcastReceiver> receivers = new ArrayList<BroadcastReceiver>();
private static ReceiverManager ref;
private Context context;
private ReceiverManager(Context context){
this.context = context;
}
public static synchronized ReceiverManager init(Context context) {
if (ref == null) ref = new ReceiverManager(context);
return ref;
}
public Intent registerReceiver(BroadcastReceiver receiver, IntentFilter intentFilter){
receivers.add(receiver);
Intent intent = context.registerReceiver(receiver, intentFilter);
Log.i(getClass().getSimpleName(), "registered receiver: "+receiver+" with filter: "+intentFilter);
Log.i(getClass().getSimpleName(), "receiver Intent: "+intent);
return intent;
}
public boolean isReceiverRegistered(BroadcastReceiver receiver){
boolean registered = receivers.contains(receiver);
Log.i(getClass().getSimpleName(), "is receiver "+receiver+" registered? "+registered);
return registered;
}
public void unregisterReceiver(BroadcastReceiver receiver){
if (isReceiverRegistered(receiver)){
receivers.remove(receiver);
context.unregisterReceiver(receiver);
Log.i(getClass().getSimpleName(), "unregistered receiver: "+receiver);
}
}
}
आपके पास कई विकल्प हैं
आप अपनी कक्षा या गतिविधि में एक झंडा लगा सकते हैं। अपनी कक्षा में एक बूलियन चर डालें और इस झंडे को देखें ताकि यह पता चल सके कि आपके पास प्राप्तकर्ता पंजीकृत है या नहीं।
एक वर्ग बनाएं जो रिसीवर को बढ़ाता है और वहां आप उपयोग कर सकते हैं:
आपकी परियोजना में इस वर्ग का केवल एक उदाहरण के लिए सिंगलटन पैटर्न है।
यदि प्राप्तकर्ता पंजीकृत है, तो यह जानने के लिए तरीकों को लागू करें।
आपको कोशिश / पकड़ का उपयोग करना होगा:
try {
if (receiver!=null) {
Activity.this.unregisterReceiver(receiver);
}
} catch (IllegalArgumentException e) {
e.printStackTrace();
}
आप इसे आसान कर सकते हैं ...।
1) एक बूलियन वेरिएबल बनाएं ...
private boolean bolBroacastRegistred;
2) जब आप अपने ब्रॉडकास्टर रिसीवर को पंजीकृत करते हैं, तो इसे TRUE पर सेट करें
...
bolBroacastRegistred = true;
this.registerReceiver(mReceiver, new IntentFilter(BluetoothDevice.ACTION_FOUND));
....
3) onPause () में इसे करें ...
if (bolBroacastRegistred) {
this.unregisterReceiver(mReceiver);
bolBroacastRegistred = false
}
बस यह, और अब, आपको ऑनपॉज़ () पर अधिक अपवाद त्रुटि संदेश नहीं मिलेगा।
टिप 1: हमेशा ऑनग्रेज़र () ऑनपॉज़ में उपयोग न करें
सफलता!
मैंने प्रसारण प्राप्तकर्ता को मुख्य गतिविधि थ्रेड के हैंडलर उदाहरण के बारे में बताने के लिए आशय का उपयोग किया और संदेश का उपयोग मुख्य गतिविधि को संदेश देने के लिए किया
मैंने इस तरह के तंत्र का उपयोग यह जांचने के लिए किया है कि ब्रॉडकास्ट रिसीवर पहले से पंजीकृत है या नहीं। कभी-कभी इसकी आवश्यकता तब होती है जब आप अपने प्रसारण रिसीवर को गतिशील रूप से पंजीकृत करते हैं और दो बार या आप उपयोगकर्ता को प्रस्तुत करना नहीं चाहते हैं यदि प्रसारण रिसीवर चल रहा है।
मुख्य गतिविधि:
public class Example extends Activity {
private BroadCastReceiver_example br_exemple;
final Messenger mMessenger = new Messenger(new IncomingHandler());
private boolean running = false;
static class IncomingHandler extends Handler {
@Override
public void handleMessage(Message msg) {
running = false;
switch (msg.what) {
case BroadCastReceiver_example.ALIVE:
running = true;
....
break;
default:
super.handleMessage(msg);
}
}
}
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
IntentFilter filter = new IntentFilter();
filter.addAction("pl.example.CHECK_RECEIVER");
br_exemple = new BroadCastReceiver_example();
getApplicationContext().registerReceiver(br_exemple , filter); //register the Receiver
}
// call it whenever you want to check if Broadcast Receiver is running.
private void check_broadcastRunning() {
/**
* checkBroadcastHandler - the handler will start runnable which will check if Broadcast Receiver is running
*/
Handler checkBroadcastHandler = null;
/**
* checkBroadcastRunnable - the runnable which will check if Broadcast Receiver is running
*/
Runnable checkBroadcastRunnable = null;
Intent checkBroadCastState = new Intent();
checkBroadCastState .setAction("pl.example.CHECK_RECEIVER");
checkBroadCastState .putExtra("mainView", mMessenger);
this.sendBroadcast(checkBroadCastState );
Log.d(TAG,"check if broadcast is running");
checkBroadcastHandler = new Handler();
checkBroadcastRunnable = new Runnable(){
public void run(){
if (running == true) {
Log.d(TAG,"broadcast is running");
}
else {
Log.d(TAG,"broadcast is not running");
}
}
};
checkBroadcastHandler.postDelayed(checkBroadcastRunnable,100);
return;
}
.............
}
प्रसारण प्राप्तकर्ता:
public class BroadCastReceiver_example extends BroadcastReceiver {
public static final int ALIVE = 1;
@Override
public void onReceive(Context context, Intent intent) {
// TODO Auto-generated method stub
Bundle extras = intent.getExtras();
String action = intent.getAction();
if (action.equals("pl.example.CHECK_RECEIVER")) {
Log.d(TAG, "Received broadcast live checker");
Messenger mainAppMessanger = (Messenger) extras.get("mainView");
try {
mainAppMessanger.send(Message.obtain(null, ALIVE));
} catch (RemoteException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
.........
}
}
व्यक्तिगत रूप से मैं unregisterReceiver को कॉल करने और अपवाद को निगलने की विधि का उपयोग करता हूं यदि इसे फेंक दिया जाता है। मैं मानता हूं कि यह बदसूरत है लेकिन वर्तमान में प्रदान की गई सबसे अच्छी विधि है।
मैंने यह देखने के लिए बूलियन पद्धति प्राप्त करने के लिए एक सुविधा अनुरोध उठाया है कि क्या कोई रिसीवर एंड्रॉइड एपीआई में जोड़ा गया है। अगर आप इसे जोड़ना चाहते हैं तो कृपया इसे यहाँ सपोर्ट करें: https://code.google.com/p/android/issues/detail?id=71818
मुझे आपकी समस्या है, मैंने अपने आवेदन में उसी समस्या का सामना किया। मैं आवेदन के भीतर कई बार registerReceiver () कॉल कर रहा था।
इस समस्या का एक सरल समाधान है कि आप अपने कस्टम एप्लिकेशन क्लास में रजिस्टरराइवर () को कॉल करें। यह सुनिश्चित करेगा कि आपके ब्रॉडकास्ट रिसीवर को आपके पूरे एप्लिकेशन जीवनचक्र में केवल एक ही कहा जाएगा।
public class YourApplication extends Application
{
@Override
public void onCreate()
{
super.onCreate();
//register your Broadcast receiver here
IntentFilter intentFilter = new IntentFilter("MANUAL_BROADCAST_RECIEVER");
registerReceiver(new BroadcastReciever(), intentFilter);
}
}
मैंने इसे कैसे किया है, यह ceph3us द्वारा दिए गए उत्तर का संशोधित संस्करण है और इसे slinden77 द्वारा संपादित किया गया है (अन्य बातों के अलावा मैंने उन तरीकों के रिटर्न मानों को हटा दिया है जिनकी मुझे आवश्यकता नहीं थी):
public class MyBroadcastReceiver extends BroadcastReceiver{
private boolean isRegistered;
public void register(final Context context) {
if (!isRegistered){
Log.d(this.toString(), " going to register this broadcast receiver");
context.registerReceiver(this, new IntentFilter("MY_ACTION"));
isRegistered = true;
}
}
public void unregister(final Context context) {
if (isRegistered) {
Log.d(this.toString(), " going to unregister this broadcast receiver");
context.unregisterReceiver(this);
isRegistered = false;
}
}
@Override
public void onReceive(final Context context, final Intent intent) {
switch (getResultCode()){
//DO STUFF
}
}
}
फिर एक गतिविधि वर्ग पर:
public class MyFragmentActivity extends SingleFragmentActivity{
MyBroadcastReceiver myBroadcastReceiver;
@Override
protected void onCreate(Bundle savedInstanceState){
super.onCreate(savedInstanceState);
registerBroacastReceiver();
}
@Override
protected Fragment createFragment(){
return new MyFragment();
}
//This method is called by the fragment which is started by this activity,
//when the Fragment is done, we also register the receiver here (if required)
@Override
public void receiveDataFromFragment(MyData data) {
registerBroacastReceiver();
//Do some stuff
}
@Override
protected void onStop(){
unregisterBroacastReceiver();
super.onStop();
}
void registerBroacastReceiver(){
if (myBroadcastReceiver == null)
myBroadcastReceiver = new MyBroadcastReceiver();
myBroadcastReceiver.register(this.getApplicationContext());
}
void unregisterReceiver(){
if (MyBroadcastReceiver != null)
myBroadcastReceiver.unregister(this.getApplicationContext());
}
}
मैंने इस कोड को अपनी मूल गतिविधि में डाल दिया है
सूचीबध्द सूची = नया एरियर सूची <> ();
@Override
public Intent registerReceiver(BroadcastReceiver receiver, IntentFilter filter) {
registeredReceivers.add(System.identityHashCode(receiver));
return super.registerReceiver(receiver, filter);
}
@Override
public void unregisterReceiver(BroadcastReceiver receiver) {
if(registeredReceivers.contains(System.identityHashCode(receiver)))
super.unregisterReceiver(receiver);
}
मेरे लिए निम्नलिखित काम किया:
if (receiver.isOrderedBroadcast()) {
requireContext().unregisterReceiver(receiver);
}
यहां मैंने यह जांचने के लिए किया था कि क्या ब्रॉडकास्टर पहले से पंजीकृत है, भले ही आप आवेदन बंद कर दें (खत्म करें)
फ़र्स्टटाइम आपके एप्लिकेशन को चलाने वाला, पहले प्रसारण भेजें, यह सही / गलत होगा, यह इस बात पर निर्भर करता है कि आपका ब्रॉडकास्टर अभी भी चल रहा है या नहीं।
माय ब्रॉडकास्टर
public class NotificationReceiver extends BroadcastReceiver {
@Override
public void onReceive(Context context, Intent intent) {
if(intent.getExtras() != null && intent.getStringExtra("test") != null){
Log.d("onReceive","test");
return;
}
}
}
मेरी मुख्यता
// init Broadcaster
private NotificationReceiver nr = new NotificationReceiver();
Intent msgrcv = new Intent("Msg");
msgrcv.putExtra("test", "testing");
boolean isRegistered = LocalBroadcastManager.getInstance(this).sendBroadcast(msgrcv);
if(!isRegistered){
Toast.makeText(this,"Starting Notification Receiver...",Toast.LENGTH_LONG).show();
LocalBroadcastManager.getInstance(this).registerReceiver(nr,new IntentFilter("Msg"));
}
आप उस रिसीवर का संदर्भ बनाने के लिए डैगर का उपयोग कर सकते हैं ।
पहले इसे प्रदान करें:
@Provides
@YourScope
fun providesReceiver(): NotificationReceiver{
return NotificationReceiver()
}
फिर उसे इंजेक्ट करें जहाँ आपको ज़रूरत हो (उपयोग constructor
या फ़ील्ड injection
)
और बस इसे पास करें registerReceiver
।
इसे try/catch
ब्लॉक में भी रखें।
if( receiver.isOrderedBroadcast() ){
// receiver object is registered
}
else{
// receiver object is not registered
}
बस NullPointerException की जाँच करें। यदि रिसीवर मौजूद नहीं है, तो ...
try{
Intent i = new Intent();
i.setAction("ir.sss.smsREC");
context.sendBroadcast(i);
Log.i("...","broadcast sent");
}
catch (NullPointerException e)
{
e.getMessage();
}