从广播接收器调用活动方法

17
在主Activity中,加载了一个包含一些输入字段和提交按钮的布局。当单击提交按钮时,onClick处理程序方法将短信发送回相同的手机号码:

在主活动中,加载了一个包含一些输入字段和提交按钮的布局。当单击提交按钮时,onClick处理程序方法将短信发送回相同的手机号码:

SmsManager sms = SmsManager.getDefault();
sms.sendTextMessage(number, null, "hi", null, null);

已定义了一个广播接收器,拦截消息:

public class SmsReceiver extends BroadcastReceiver {

@Override
public void onReceive(Context context, Intent intent) {
    Bundle pdusBundle = intent.getExtras();
    Object[] pdus=(Object[])pdusBundle.get("pdus");
    SmsMessage messages=SmsMessage.createFromPdu((byte[]) pdus[0]);
    if(messages.getMessageBody().contains("hi")){
        abortBroadcast();
    }

}
}

现在,我想从广播接收器调用我的主活动中的一个带有参数的函数。是否可能?如果是,我应该在我的广播接收器中添加什么样的代码?


2
这个应该能帮到你:https://dev59.com/fnLYa4cB1Zd3GeqPdfKB - Manishika
5个回答

24

感谢@Manishika。为了进一步说明,将Broadcastreceiver设置为动态的,而不是在清单文件中定义,就解决了问题。因此,在我的广播接收器类中,我添加了以下代码:

MainActivity main = null;
void setMainActivityHandler(MainActivity main){
    this.main=main;
}
在BroadcastReceiver类的onReceive函数的结尾,我调用了主Activity的一个函数:
main.verifyPhoneNumber("hi");

在主活动中,在发送短信之前,我动态地定义和注册广播接收器:

SmsReceiver BR_smsreceiver = null;
BR_smsreceiver = new SmsReceiver();
BR_smsreceiver.setMainActivityHandler(this);
IntentFilter fltr_smsreceived = new IntentFilter("android.provider.Telephony.SMS_RECEIVED");
registerReceiver(BR_smsreceiver,fltr_smsreceived);  

你好@faizal,我需要让我的BroadcastReceiver一直运行,即使我的应用程序不可见。我能够通过menifext.xml文件中的配置来实现这一点。我还需要从我的BroadcastReceiver调用我的Activity方法。我能够使用您上面提供的解决方案调用我的Activity方法,但是这样做会停止我的BroadcastReceiver,一旦我退出我的应用程序。而且,使用我的自己的配置,我的BroadcastReceiver正在运行,但我无法调用我的Activity方法。有什么办法可以同时完成这两个任务吗?谢谢,Jai。 - Jaikrat

8
将您的Activity上下文传递给BroadcastReceiver的构造函数。
public class SmsReceiver extends BroadcastReceiver{

    MainActivity ma; //a reference to activity's context

    public SmsReceiver(MainActivity maContext){
        ma=maContext;
    }

    @Override
    public void onReceive(Context context, Intent intent) {
        ma.brCallback("your string"); //calling activity method
    }

}

在你的MainActivity中

public class MainActivity extends AppCompatActivity {
    ...
    public void onStart(){
        ...        
    SmsReceiver smsReceiver = new SmsReceiver(this); //passing context
    LocalBroadcastManager.getInstance(this).registerReceiver(smsReceiver,null);
        ...
    }

    public void brCallback(String param){
        Log.d("BroadcastReceiver",param);
    }
}

希望您能获得帮助。

LocalBroadcastManager现在已经不起作用了,我不得不使用IntentFilter filter = new IntentFilter(); filter.addAction("com.google.android.gms.auth.api.phone.SMS_RETRIEVED"); this.registerReceiver(smsReceiver, filter); - mikep

2

使用这个

Intent intent=new Intent();
intent.setClassName("com.package.my", "bcd.class");
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);

context.startActivity(intent);

1
我不能使用你的解决方案,因为我不想启动一个新的主活动。我已经发布了一个带有解决方案的答案。 - faizal

0

除非它是公共静态方法,否则您无法直接在Activity中调用函数,但不要这样做。我建议这样做:

@Override
public void onReceive(Context context, Intent intent) {
    Bundle bundle = new Bundle();
    bundle.putExtraString("ACTION", "stopBroadcast");

    // ### put what ever you need into the bundle here ###

    Intent intent = new Intent();
    intent.setClassName(context, "activity.class");
    intent.putExtras(bundle);

    context.startActivity(intent);
}

然后从您的Activity的onCreate()获取Bundle并根据需要执行操作。


1
是的,你说得对。我主活动中需要调用的函数必须是公共的。我不能使用你的解决方案,因为我不想启动一个新的主活动。我已经发布了一个带有解决方案的答案。 - faizal

-4

有点晚了,但是没有人提到过。 不需要传递活动。 正如@faizal所指定的那样,IntentService是从MainActivity启动的,因此onReceive()中的context已经是MainActivity的实例。 只需编写类似于以下内容的代码:

@Override
public void onReceive(Context context, Intent intent) {
    if(context instanceof MainActivity) {
        MainActivity activity = (MainActivity) context;
    }
}

1
java.lang.ClassCastException: android.app.ReceiverRestrictedContext 无法转换为 com.mobiquel.udhampur.ui.home.HomeActivity。这将导致应用程序崩溃。 - Arman Reyaz

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接