利用 Handler 在不是内部类的 BroadcastReceiver 中更新 UI
我们知道 Receiver 也是在主线程中运行的,如果将 Receiver 写成 activity 的内部类,可以直接在 onReceive()方法中获得主线程的 UI 控件,对 UI 进行刷新,但是如果我们将 Receiver 写成单独的一个类,那这样就可能比较麻烦了,因为我们获取 activity 中的 UI 控件就变得困难。
就算我们利用 LayoutInflater.from(C).inflate(R.layout.mainframe,null)方法来获得布局和控件,但是这不是取得你的 Context 中的控件,而是将 mainframe.xml 中的控件创建出来一份,与原本的 Context 无关。所以这时候我就想到了利用 handler 来传递信息,在主线程中更新 UI。
我的目标是,在 activity 中发送一个广播,Receiver 收到广播后,发送一个 message,handler 收到这个消息后,通过 handlemessage()方法对 UI 进行刷新,下面是我的源码:
首先是界面的布局,在界面上就比较简单,点击按钮发送一个广播,然后 textview 的内容发生改变:
<Button
android:id="@+id/send_broadcast"
android:text="send_broadcast"
android:onClick="sendBroadcast"
android:layout_width="match_parent"
android:layout_height="wrap_content" />
<TextView
android:id="@+id/receiver_text"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:textSize="20dp"
android:text="00000"
/>
然后注册我们自己的 Receiver,为了获得 activity 中的 handler,提供一个构造方法,将 handler 传递过来。收到广播后通过 handler 发送 mesage:
public MyReceiver(Handler handler){
this.handler=handler;
}
@Override
public void onReceive(Context context, Intent intent) {
// TODO: This method is called when the BroadcastReceiver is receiving
// an Intent broadcast.
Message message=new Message();
message.what=1;
message.obj="收到了广播";
handler.sendMessage(message);
Log.d(TAG, "onReceive: ++++++++++++");
}
最后是 mainactivity,在 mainactivity 中发送广播,创建 handler 实例,收到 message 后,更新 UI:
/*创建handler,在handlemessage中进行UI操作*/
private TextView receiverText;
private MyReceiver receiver;
private String TAG="mainactivity";
private Handler handler=new Handler(){
@Override
public void handleMessage(@NonNull Message msg) {
super.handleMessage(msg);
Log.d(TAG, "handleMessage:+++++++++ ");
switch (msg.what){
case 1:
Log.d(TAG, "handleMessage:+++++++++ ");
receiverText.setText((String)msg.obj);
}
}
};
/*动态注册广播*/
IntentFilter filter=new IntentFilter("com.example.receiver");
receiver=new MyReceiver(handler);
registerReceiver(receiver,filter);
/* 发送广播*/
public void sendBroadcast(View v){
Intent intent=new Intent("com.example.receiver");
sendBroadcast(intent);
}
最后的效果如图所示: