登録した受信者がまだ登録されているかどうかを確認する必要があります。登録されていない場合、どのように確認すればよいですか?
17 に答える
受信者が登録されているかどうかを確認するAPI関数はありません。回避策は、コードをtry catch block as done below.
try {
//Register or UnRegister your broadcast receiver here
} catch(IllegalArgumentException e) {
e.printStackTrace();
}
このスレッドを検討すると、APIが直接APIを提供するかどうかはわかりません。
私は同じことを考えていました。
私の場合、受け取ったインテントを処理した後、引数として自分自身を渡すことBroadcastReceiver
を呼び出す実装が あります。 複数のに登録されているため、レシーバーのメソッドが複数回呼び出される可能性がわずかにあり、からスローされる可能性があります。Context#unregisterReceiver(BroadcastReceiver)
onReceive(Context, Intent)
IntentFilters
IllegalArgumentException
Context#unregisterReceiver(BroadcastReceiver)
私の場合、呼び出す前にチェックするプライベート同期メンバーを保存でき
Context#unregisterReceiver(BroadcastReceiver)
ますが、APIがチェックメソッドを提供する方がはるかにクリーンです。
最も簡単な解決策
受信機で:
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フラグを設定することを忘れないようにする必要があるため、これは実際にはそれほどエレガントではありません。–ステルスラビ
-「よりエレガントな方法」は、フラグを登録および設定するための受信機にメソッドを追加しました
デバイスを再起動した場合、またはアプリがOSによって強制終了された場合、これは機能しません。–アミン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を拡張するクラスを作成すると、次のものを使用できます。
プロジェクトにこのクラスのインスタンスが1つしかないシングルトンパターン。
受信者が登録されているかどうかを知るためのメソッドを実装します。
try/catchを使用する必要があります。
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
}
それだけで、onPause()でこれ以上例外エラーメッセージが表示されることはありません。
ヒント1:onDestroy()ではなくonPause()で常にunregisterReceiver()を使用します。ヒント2:unregisterReceive()を実行するときは、bolBroadcastRegistred変数をFALSEに設定することを忘れないでください。
成功!
これをonDestroyまたはonStopメソッドに配置した場合。アクティビティが再度作成されたとき、MessageReciverは作成されていなかったと思います。
@Override
public void onDestroy (){
super.onDestroy();
LocalBroadcastManager.getInstance(context).unregisterReceiver(mMessageReceiver);
}
インテントを使用してブロードキャストレシーバーにメインアクティビティスレッドのハンドラーインスタンスを通知し、メッセージを使用してメインアクティビティにメッセージを渡しました
私はこのようなメカニズムを使用して、BroadcastReceiverがすでに登録されているかどうかを確認しました。Broadcast Receiverを動的に登録し、2回登録したくない場合や、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を呼び出し、例外がスローされた場合に例外を飲み込むメソッドを使用します。私はこれが醜いことに同意しますが、現在提供されている最良の方法です。
レシーバーがAndroidAPIに追加されて登録されているかどうかを確認するブールメソッドを取得する機能リクエストを発生させました。追加されたものを確認したい場合は、ここでサポートしてください: https ://code.google.com/p/android/issues/detail?id=73718
私はあなたの問題を抱えています、私は私のアプリケーションで同じ問題に直面しました。アプリケーション内でregisterReceiver()を複数回呼び出していました。
この問題の簡単な解決策は、カスタムアプリケーションクラスでregisterReceiver()を呼び出すことです。これにより、ブロードキャストレシーバーがアプリケーションライフサイクル全体で1つだけ呼び出されるようになります。
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());
}
}
私はこのコードを私の親の活動に入れました
リストregisteredReceivers=new 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);
}
私にとって、以下はうまくいきました:
if (receiver.isOrderedBroadcast()) {
requireContext().unregisterReceiver(receiver);
}
アプリケーションを閉じても、Broadcasterがすでに登録されているかどうかを確認するために私が行ったことは次のとおりです(finish())
アプリケーションを実行しているFirstimeは、最初にブロードキャストを送信します。ブロードキャストがまだ実行されているかどうかによって、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;
}
}
}
私の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"));
}
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();
}