Intereting Posts
Как создать список с таблицами данных как элементы, которые можно сортировать Android: передать событие Touch to underlaying view Получение NoClassDefFoundError: android.os.AsyncTask Как установить APK-файл в эмулятор Android? Нестатический метод 'getSharedPreferences (java.lang.String, int)' не может ссылаться на статический контекст Почему код из учебников Android SDK не работает? Список выбора отключен в AutoCompleteTextView в диалоге Spinner не применяет атрибут dropDownSelector Как я могу отобразить индикатор выполнения при копировании файлов с помощью Apache FileUtils? Android FTPClient Uploading File = SocketException: сброс соединения Правильный способ использования ТОЛЬКО гироскопа и акселерометра для получения надежного токового угла на любой оси на ANDROID Google Analytics – исключение NullPointerException в тесте Как сделать макет прокручиваемым? Android Java.lang.IllegalArgumentException: просмотр не подключен к менеджеру окон Как добавить ярлык приложения на главный экран

Как проверить, зарегистрирован ли приемник в Android?

Мне нужно проверить, зарегистрирован ли зарегистрированный ресивер, если нет, как я могу проверить какие-либо методы?

Я не уверен, что API предоставляет напрямую API, если вы рассматриваете этот поток :

Я думал о том же самом.
В моем случае у меня есть реализация BroadcastReceiver которая вызывает Context#unregisterReceiver(BroadcastReceiver) передающую себя как arg после обработки намерения, которое оно получает.
Существует небольшая вероятность того, что метод onReceive(Context, Intent) получателя onReceive(Context, Intent) более одного раза, поскольку он зарегистрирован несколькими IntentFilters , создавая потенциал IntentFilters IllegalArgumentException из IntentFilters Context#unregisterReceiver(BroadcastReceiver) .

В моем случае я могу сохранить частный синхронизированный элемент для проверки перед вызовом Context#unregisterReceiver(BroadcastReceiver) , но было бы намного чище, если API предоставил метод проверки.

Функция API не проверяет, зарегистрирован ли приемник. Обходной путь состоит в том, чтобы поместить ваш код в блок try {...} catch(IllegalArgumentException e) {...} .

Я использую это решение

 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); } } } 

У вас есть несколько вариантов

  1. Вы можете поместить флаг в свой класс или деятельность. Поместите логическую переменную в ваш класс и посмотрите на этот флаг, чтобы узнать, зарегистрирован ли получатель.

  2. Создайте класс, который расширяет приемник и там вы можете использовать:

    1. Шаблон Singleton имеет только один экземпляр этого класса в вашем проекте.

    2. Внедрите методы, чтобы узнать, зарегистрирован ли приемник.

Простейшее решение

В приемнике:

 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.registerReceiver(Conext, IntentFilter); // register myReceiver.unregister(Context); // unregister 

Изменить I

— в ответ на

Это действительно не так элегантно, потому что вы должны помнить, чтобы установить флаг isRegistered после регистрации. – Стелс Раввин

– «более ellegant way» добавил метод в приемнике для регистрации и установки флага

Если вы положите это на метод onDestroy или onStop. Я думаю, что когда действие было создано снова, MessageReciver не создавался.

 @Override public void onDestroy (){ super.onDestroy(); LocalBroadcastManager.getInstance(context).unregisterReceiver(mMessageReceiver); } 

Вы можете сделать это легко ….

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 } 

Только это, и теперь вы не получите больше сообщений об ошибке исключения onPause ().

Tip1: Всегда используйте unregisterReceiver () в onPause () не в onDestroy () Tip2: не забудьте установить переменную bolBroadcastRegistred в FALSE при запуске unregisterReceive ()

Успех!

Вы должны использовать try / catch:

 try { if (receiver!=null) { Activity.this.unregisterReceiver(receiver); } } catch (IllegalArgumentException e) { e.printStackTrace(); } 

Я использовал Intent, чтобы позволить Broadcast Receiver узнать о экземпляре Handler основного потока активности и использовать Message для передачи сообщения в Main activity

Я использовал такой механизм, чтобы проверить, зарегистрирован ли широковещательный приемник или нет. Иногда это необходимо, когда вы регистрируете свой широковещательный приемник динамически и не хотите делать его дважды или вы представляете пользователю, если работает широковещательный приемник.

Основная деятельность:

 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 и проглатывает исключение, если оно выбрано. Я согласен, что это уродливый, но лучший метод, который в настоящее время предоставляется.

Я поднял запрос функции, чтобы получить логический метод, чтобы проверить, зарегистрирован ли ресивер в Android API. Пожалуйста, поддержите его здесь, если хотите добавить его: https://code.google.com/p/android/issues/detail?id=73718

Вот что я сделал, чтобы проверить, зарегистрирован ли Broadcaster, даже если вы закрываете приложение (finish ())

Сначала запускайте приложение, сначала отправляйте широковещательную передачу, и оно вернет true / false, зависит от того, работает ли ваш вещатель в режиме ожидания или нет.

Мой транслятор

 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")); } 

Вот как я это сделал, это модифицированная версия ответа, данная 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 } } } 

Затем в классе Activity:

 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()); } } 
 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(); }