Мне нужно проверить, зарегистрирован ли мой зарегистрированный получатель, если нет, как мне проверить это какими-либо методами?
Как проверить, зарегистрирован ли Receiver в Android?
Ответы (18)
Я не уверен, что API предоставляет API напрямую, если вы рассматриваете этот поток < / а>:
Мне было интересно то же самое.
В моем случае у меня есть реализацияBroadcastReceiver
, которая вызываетContext#unregisterReceiver(BroadcastReceiver)
, передавая себя в качестве аргумента после обработки полученного намерения.
Существует небольшая вероятность того, что методonReceive(Context, Intent)
получателя будет вызван more более одного раза, так как он зарегистрирован с несколькимиIntentFilters
, что создает вероятность выбросаIllegalArgumentException
изContext#unregisterReceiver(BroadcastReceiver)
.В моем случае я могу сохранить частный синхронизированный член для проверки перед вызовом
Context#unregisterReceiver(BroadcastReceiver)
, но было бы намного чище, если бы API предоставлял метод проверки.
Нет функции API, чтобы проверить, зарегистрирован ли получатель. Обходной путь - поместить ваш код в try catch block as done below.
try {
//Register or UnRegister your broadcast receiver here
} catch(IllegalArgumentException e) {
e.printStackTrace();
}
null
. Но, как вы отметили, я собираюсь использовать try catch
. Нелепый.
- person ; 28.11.2012
простейшее решение
в приемнике:
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
объявление 1
-- в ответ на:
Это действительно не так элегантно, потому что вы должны не забыть установить флаг isRegistered после регистрации. - Стелс раввин
- "более элегантный способ" добавлен метод в приемнике для регистрации и установки флага
это не сработает, если вы перезагрузите устройство или ваше приложение было убито ОС. - amin 6 часов назад
@amin - увидеть время жизни зарегистрированного получателя в коде (не зарегистрированном системой в манифесте) :)
Я использую это решение
public class ReceiverManager {
private WeakReference<Context> cReference;
private static List<BroadcastReceiver> receivers = new ArrayList<BroadcastReceiver>();
private static ReceiverManager ref;
private ReceiverManager(Context context) {
cReference = new WeakReference<>(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 = cReference.get().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);
cReference.get().unregisterReceiver(receiver);
Log.i(getClass().getSimpleName(), "unregistered receiver: " + receiver);
}
}
}
У вас есть несколько вариантов
Вы можете поставить отметку в своем классе или мероприятии. Поместите логическую переменную в свой класс и посмотрите на этот флаг, чтобы узнать, зарегистрирован ли у вас Receiver.
Создайте класс, расширяющий Receiver, и в нем вы можете использовать:
Шаблон Singleton для того, чтобы иметь только один экземпляр этого класса в вашем проекте.
Реализуйте методы, чтобы узнать, зарегистрирован ли получатель.
Вы должны использовать try / catch:
try {
if (receiver!=null) {
Activity.this.unregisterReceiver(receiver);
}
} catch (IllegalArgumentException e) {
e.printStackTrace();
}
Вы можете сделать это легко ....
1) создать логическую переменную ...
private boolean bolBroacastRegistred;
2) Когда вы регистрируете свой широковещательный приемник, установите для него значение ИСТИНА.
...
bolBroacastRegistred = true;
this.registerReceiver(mReceiver, new IntentFilter(BluetoothDevice.ACTION_FOUND));
....
3) В onPause () сделайте это ...
if (bolBroacastRegistred) {
this.unregisterReceiver(mReceiver);
bolBroacastRegistred = false
}
Только это, и теперь вы больше не будете получать сообщения об ошибке исключения в onPause ().
Совет 1. Всегда используйте unregisterReceiver () в onPause (), а не в onDestroy (). Совет 2. Не забудьте установить для переменной bolBroadcastRegistred значение FALSE при запуске unregisterReceive ().
Успех!
Если вы поставите это на метод onDestroy или onStop. Я думаю, что когда действие было создано снова, MessageReciver не создавался.
@Override
public void onDestroy (){
super.onDestroy();
LocalBroadcastManager.getInstance(context).unregisterReceiver(mMessageReceiver);
}
Я использовал намерение, чтобы широковещательный приемник знал об экземпляре обработчика основного потока активности, и использовал сообщение, чтобы передать сообщение в основное действие.
Я использовал такой механизм, чтобы проверить, зарегистрирован ли Broadcast Receiver или нет. Иногда это необходимо, когда вы динамически регистрируете свой широковещательный приемник и не хотите делать это дважды, или когда вы показываете пользователю, что широковещательный приемник запущен.
Основная деятельность:
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.
Я понял вашу проблему, я столкнулся с той же проблемой в своем приложении. Я несколько раз звонил registerReceiver () в приложении.
Простое решение этой проблемы - вызвать 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);
}
}
Для меня сработало следующее:
if (receiver.isOrderedBroadcast()) {
requireContext().unregisterReceiver(receiver);
}
Вот как я это сделал, это измененная версия ответа, данного 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());
}
}
я поместил этот код в свою родительскую активность
Список зарегистрированных получателей = новый ArrayList ‹> ();
@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);
}
Вот что я сделал, чтобы проверить, зарегистрирован ли уже Broadcaster, даже если вы закроете приложение (finish ())
При первом запуске вашего приложения отправьте сначала широковещательную рассылку, она вернет истину / ложь, в зависимости от того, работает ваш вещатель или нет.
Мой телеведущий
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"));
}
Вы можете использовать Dagger, чтобы создать ссылку на этого получателя.
Сначала предоставьте это:
@Provides
@YourScope
fun providesReceiver(): NotificationReceiver{
return NotificationReceiver()
}
Затем введите его туда, где вам нужно (используя constructor
или поле injection
)
и просто передайте его registerReceiver
.
Также поместите его в блок try/catch
.
Просто проверьте NullPointerException. Если ресивер не существует, то ...
try{
Intent i = new Intent();
i.setAction("ir.sss.smsREC");
context.sendBroadcast(i);
Log.i("...","broadcast sent");
}
catch (NullPointerException e)
{
e.getMessage();
}