Мне нужно проверить, зарегистрирован ли зарегистрированный ресивер, если нет, как проверить его какие-либо методы?
Как проверить, зарегистрирован ли приемник на Android?
Ответ 1
Я не уверен, что API предоставляет непосредственно API, если вы рассматриваете эту тему:
Я думал о том же самом.
В моем случае у меня есть реализацияBroadcastReceiver
которая вызываетContext#unregisterReceiver(BroadcastReceiver)
передавая себя в качестве аргумента после обработки получаемого намерения.
Существует небольшая вероятность того, что метод приемникаonReceive(Context, Intent)
вызывается более одного раза, так как он зарегистрирован в несколькихIntentFilters
, создавая возможность дляIntentFilters
IllegalArgumentException
изContext#unregisterReceiver(BroadcastReceiver)
.В моем случае я могу сохранить частный синхронизированный элемент для проверки перед вызовом
Context#unregisterReceiver(BroadcastReceiver)
, но было бы намного чище, если бы API предоставил метод проверки.
Ответ 2
Нет функции API, чтобы проверить, зарегистрирован ли получатель. Обходной путь - поместить ваш код в блок try catch block as done below.
try {
//Register or UnRegister your broadcast receiver here
} catch(IllegalArgumentException e) {
e.printStackTrace();
}
Ответ 3
простейшее решение
в приемнике:
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 флаг после регистрации. - Раввин Стелс
- "более элегантный способ" добавлен метод в приемнике для регистрации и установки флага
это не будет работать, если вы перезагрузите устройство или если ваше приложение было убито ОПЕРАЦИОННЫЕ СИСТЕМЫ. - Амин 6 часов назад
@amin - видеть время жизни зарегистрированного получателя в коде (а не в системе, зарегистрированной в декларации) :)
Ответ 4
Я использую это решение
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);
}
}
}
Ответ 5
У вас есть несколько вариантов
-
Вы можете поместить флаг в свой класс или активность. Поместите логическую переменную в ваш класс и посмотрите на этот флаг, чтобы узнать, зарегистрирован ли ваш ресивер.
-
Создайте класс, который расширяет приемник и там вы можете использовать:
-
Шаблон Singleton имеет только один экземпляр этого класса в вашем проекте.
-
Внедрить методы, чтобы знать, зарегистрирован ли приемник.
-
Ответ 6
Вам нужно использовать try/catch:
try {
if (receiver!=null) {
Activity.this.unregisterReceiver(receiver);
}
} catch (IllegalArgumentException e) {
e.printStackTrace();
}
Ответ 7
Вы можете сделать это легко....
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().
Tip1: Всегда используйте unregisterReceiver() в onPause() не в onDestroy() Tip2: Не забудьте установить переменную bolBroadcastRegistred в FALSE при запуске unregisterReceive()
Успех!
Ответ 8
Если вы положите это на метод onDestroy или onStop. Я думаю, что когда действие было создано снова, MessageReciver не создавался.
@Override
public void onDestroy (){
super.onDestroy();
LocalBroadcastManager.getInstance(context).unregisterReceiver(mMessageReceiver);
}
Ответ 9
Я использовал Intent, чтобы позволить Broadcast Receiver узнать о экземпляре Handler основного потока Activity и использовать 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();
}
}
.........
}
}
Ответ 10
Лично я использую метод вызова unregisterReceiver и проглатывает исключение, если оно было выбрано. Я согласен, что это уродливый, но лучший способ, который в настоящее время предоставляется.
Я поднял запрос функции, чтобы получить логический метод, чтобы проверить, добавлен ли ресивер в API Android. Пожалуйста, поддержите его здесь, если хотите добавить его: https://code.google.com/p/android/issues/detail?id=73718
Ответ 11
Я понял вашу проблему, я столкнулся с той же проблемой в моем приложении. Я вызывал registerReceiver() несколько раз в приложении.
Простое решение этой проблемы - вызвать registerReceiver() в вашем классе пользовательских приложений. Это гарантирует, что ваш приемник Broadcast будет вызываться только один на протяжении всего жизненного цикла приложения.
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);
}
}
Ответ 12
Вот как я это сделал, это модифицированная версия ответа, данного 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());
}
}
Ответ 13
Я помещаю этот код в свою родительскую активность
Список зарегистрированныхReceivers = новый 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);
}
Ответ 14
Вот что я сделал, чтобы проверить, зарегистрирован ли 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;
}
}
}
My MainActivity
// 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"));
}
Ответ 15
Вы можете использовать Dagger, чтобы создать ссылку на этот приемник.
Сначала предоставьте это:
@Provides
@YourScope
fun providesReceiver(): NotificationReceiver{
return NotificationReceiver()
}
Затем вводят его там, где вам нужно ( с помощью constructor
или поля injection
)
и просто передайте его в registerReceiver
.
Также поместите его в блок try/catch
.
Ответ 16
if( receiver.isOrderedBroadcast() ){
// receiver object is registered
}
else{
// receiver object is not registered
}
Ответ 17
Просто проверьте исключение NullPointerException. Если приемник не существует, то...
try{
Intent i = new Intent();
i.setAction("ir.sss.smsREC");
context.sendBroadcast(i);
Log.i("...","broadcast sent");
}
catch (NullPointerException e)
{
e.getMessage();
}