登録された受信者がまだ登録されているかどうかを確認する必要があります。
このスレッド を考慮すると、APIが直接APIを提供するかどうかわかりません。
私は同じことを疑問に思っていました。
私の場合、BroadcastReceiver
の実装があり、Context#unregisterReceiver(BroadcastReceiver)
を呼び出して、受け取ったインテントを処理した後、引数として自身を渡します。
複数のIntentFilters
に登録されているため、IllegalArgumentException
がonReceive(Context, Intent)
からスローされる可能性があるため、レシーバーの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();
}
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
広告1
-への返信:
これは、登録後にisRegisteredフラグを設定することを覚えておく必要があるため、それほどエレガントではありません。 –ステルスラビ
-「より洗練された方法」は、フラグを登録および設定するためのメソッドをレシーバに追加しました
デバイスを再起動した場合、またはアプリがOSによって強制終了された場合、これは機能しません。 –アミン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);
}
}
}
いくつかのオプションがあります
クラスまたはアクティビティにフラグを付けることができます。ブール変数をクラスに入れ、このフラグを見て、Receiverが登録されているかどうかを確認します。
Receiverを拡張するクラスを作成すると、以下を使用できます。
プロジェクトのシングルトンパターンには、このクラスのインスタンスが1つしかありません。
Receiverが登録されているかどうかを知るためのメソッドを実装します。
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()を使用する
成功!
これをonDestroyまたはonStopメソッドに設定した場合。アクティビティが再び作成されたとき、MessageReciverは作成されていなかったと思います。
@Override
public void onDestroy (){
super.onDestroy();
LocalBroadcastManager.getInstance(context).unregisterReceiver(mMessageReceiver);
}
メインアクティビティスレッドのハンドラーインスタンスをブロードキャストレシーバーに知らせるためにインテントを使用し、メインアクティビティにメッセージを渡すためにメッセージを使用しました
このようなメカニズムを使用して、Broadcast Receiverが既に登録されているかどうかを確認しました。 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を呼び出して、スローされた場合に例外を飲み込むメソッドを使用します。これはいですが、現在提供されている最良の方法に同意します。
Android APIに追加されたレシーバーが登録されているかどうかを確認するためのブールメソッドを取得する機能要求を上げました。追加したい場合は、こちらでサポートしてください: 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());
}
}
このコードを親アクティビティに入れます
登録済みのReceivers = 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);
}
Dagger を使用して、そのレシーバーの参照を作成できます。
最初に提供します:
@Provides
@YourScope
fun providesReceiver(): NotificationReceiver{
return NotificationReceiver()
}
次に、必要な場所に挿入します(constructor
またはフィールドinjection
を使用)
それを単にregisterReceiver
に渡します。
また、それをtry/catch
ブロックに入れます。
以下は、アプリケーションを終了した場合でも、ブロードキャスターが既に登録されているかどうかを確認するために行ったものです(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"));
}