如何将参数传递给广播接收器的子类?

我设法让我的耳机按钮在按下时被我的应用程序识别,但其中一个按钮需要调用MyCustomActivity中的方法。问题是onReceive的第一个参数是一个无法强制转换为Active的上下文,并且使用MyCustomActivity的内部类在Android 4.1中不起作用,除非它是静态的(这具有相同的无法访问MyCustomActivity方法的问题)。

因此,留给我的唯一选择(为了同时支持2.x和4.1)是将活动作为参数传递给RemoteControlReceiver

但是我该怎么做,当实例化它的唯一方法是通过:

private ComponentName mRemoteControlReceiver = new ComponentName(this, RemoteControlReceiver.class);

哪个不接受任何其他参数?

任何想法如何解决这个限制?

注意:如果我尝试定义为具有带有参数的构造函数,则会收到以下异常:RemoteControlReceiver

E/AndroidRuntime(2836): java.lang.RuntimeException: Unable to instantiate receiver com.example.RemoteControlReceiver: java.lang.InstantiationException: can't instantiate class com.example.RemoteControlReceiver; no empty constructor

Caused by:
E/AndroidRuntime(2836): Caused by: java.lang.InstantiationException: can't instantiate class com.example.RemoteControlReceiver; no empty constructor
E/AndroidRuntime(2836):     at java.lang.Class.newInstanceImpl(Native Method)
E/AndroidRuntime(2836):     at java.lang.Class.newInstance(Class.java:1319)
E/AndroidRuntime(2836):     at android.app.ActivityThread.handleReceiver(ActivityThread.java:2205)

因此,很明显,这个新的registerMediaButtonEventReceiver要求(在Android 4.1中引入)需要一个空的构造函数

有没有办法解决这个问题?

例如,有没有办法获取对实际 RemoteControlReceiver 对象的引用(通过 间接实例化)?这样,在实例化 RemoteControlReceiver 之后,我可以使用访问器来设置它的数据成员?mAudioManager.registerMediaButtonEventReceiver()


答案 1

registerMediaButtonEventReceiver 要求在应用程序清单中声明 BroadcastReceiver。这意味着接收器必须是一个独立的类,这意味着它对您当前的活动或服务一无所知。

为了将此消息发送到您的活动或服务,您有多种选择:

  • 对活动或服务使用静态全局,以便接收方可以将消息转发给它。这通常不是一个好主意,因为它会导致泄漏,并且在以后想要更改代码时适应性不强。通常要避免静态。

  • 将消息重新广播到特定类,该类恰好是要调用的活动或服务的内部类。例如,在 BroadcastReceiver for registerMediaButtonEventReceiver 中:

    // Standalone class, declared in the manifest
    public class ButtonReceiver extends BroadcastReceiver {
        @Override
        public void onReceive(final Context context, final Intent intent) {
            Intent intent = new Intent();
            intent.setAction("com.foo.ACTION");
    
            // Rebroadcasts to your own receiver. 
            // This receiver is not exported; it'll only be received if the receiver is currently registered.
            context.sendBroadcast(intent); 
        }
    }
    

在您的活动中:

    class MyActivity extends Activity {
        private BroadcastReceiver myReceiver = new BroadcastReceiver() {
            @Override
             public void onReceive(final Context context, final Intent intent) {
                MyActivity.this.onMessageReceived();
             }
        }
        @Override
        protected void onResume() {
            registerReceiver(myReceiver, new IntentFilter("com.foo.ACTION"));
        }

        @Override
        protected void onPause() {
            unregisterReceiver(myReceiver);
        }

        private void onMessageReceived() {
        }
    }
  • 与上述方法类似,它不一定是广播,它可能是传递给活动的 Intent,具体取决于您的用例。若要执行此操作,而不是使用 sendBroadcast,请使用 startActivity(如果您使用的是服务,则使用 startService)。

答案 2

推荐