Android通過RemoteViews實(shí)現(xiàn)跨進(jìn)程更新UI示例
一、概述
前面一篇文章Android通過AIDL實(shí)現(xiàn)跨進(jìn)程更新UI我們學(xué)習(xí)了aidl跨進(jìn)程更新ui,這種傳統(tǒng)方式實(shí)現(xiàn)跨進(jìn)程更新UI是可行的,但有以下弊端:
- View中的方法數(shù)比較多,在IPC中需要增加對應(yīng)的方法比較繁瑣。
- View的每一個方法都會涉及到IPC操作,多次IPC帶來的開銷問題不容小覷。
- View中方法的某些參數(shù)可能不支持IPC傳輸。例如:OnClickListener,它僅僅是個接口沒有序列化。
接下來我們通過RemoteViews實(shí)現(xiàn)跨進(jìn)程更新UI
二、實(shí)現(xiàn)效果圖
在同一個應(yīng)用中有兩個Activity,MainActivity和Temp2Activity,這兩個Activity不在同一個進(jìn)程中。
現(xiàn)在需要通過Temp2Activity來改變MainActivity中的視圖,即在MainActivity中添加兩個Button,也就是實(shí)現(xiàn)跨進(jìn)程更新UI這么一個功能。
在MainActivity里點(diǎn)擊“跳轉(zhuǎn)到新進(jìn)程ACTIVITY”按鈕,會啟動一個新進(jìn)程的Temp2Activity,我們先點(diǎn)擊“綁定服務(wù)”,這樣我們就啟動了服務(wù),再點(diǎn)擊“AIDL更新”按鈕,通過調(diào)用handler來實(shí)現(xiàn)跨進(jìn)程更新UI,點(diǎn)擊返回,我們發(fā)現(xiàn)MainActivity頁面中新添加了兩個按鈕,并且按鈕還具有點(diǎn)擊事件。
三、核心代碼
IremoteViewsManager.aidl
里面提供了兩個方法,一個是根據(jù)id更新TextView里面的內(nèi)容,一個是根據(jù)id添加view視圖
// IremoteViewsManager.aidl.aidl package com.czhappy.remoteviewdemo; interface IremoteViewsManager { void addRemoteView(in RemoteViews remoteViews); }
RemoteViewsAIDLService.Java
package com.czhappy.remoteviewdemo.service; import android.app.Service; import android.content.Intent; import android.os.Binder; import android.os.Bundle; import android.os.IBinder; import android.os.Message; import android.os.RemoteException; import android.widget.RemoteViews; import com.czhappy.remoteviewdemo.IremoteViewsManager; import com.czhappy.remoteviewdemo.activity.MainActivity; /** * Description: * User: chenzheng * Date: 2017/2/10 0010 * Time: 10:56 */ public class RemoteViewsAIDLService extends Service { private static final String TAG = "RemoteViewsAIDLService"; private Binder remoteViewsBinder = new IremoteViewsManager.Stub(){ @Override public void addRemoteView(RemoteViews remoteViews) throws RemoteException { Message message = new Message(); message.what = 1; Bundle bundle = new Bundle(); bundle.putParcelable("remoteViews",remoteViews); message.setData(bundle); new MainActivity.MyHandler(RemoteViewsAIDLService.this,getMainLooper()).sendMessage(message); } }; public RemoteViewsAIDLService() { } @Override public IBinder onBind(Intent intent) { return remoteViewsBinder; } }
MainActivity.java
package com.czhappy.remoteviewdemo.activity; import android.content.Context; import android.content.Intent; import android.os.Bundle; import android.os.Handler; import android.os.Looper; import android.os.Message; import android.support.v7.app.AppCompatActivity; import android.util.Log; import android.view.LayoutInflater; import android.view.View; import android.widget.LinearLayout; import android.widget.RemoteViews; import android.widget.TextView; import com.czhappy.remoteviewdemo.R; import java.lang.ref.WeakReference; public class MainActivity extends AppCompatActivity { private static String TAG = "MainActivity"; private static LinearLayout mLinearLayout; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); mLinearLayout = (LinearLayout) this.findViewById(R.id.mylayout); } public static class MyHandler extends Handler { WeakReference<Context> weakReference; public MyHandler(Context context, Looper looper) { super(looper); weakReference = new WeakReference<>(context); } @Override public void handleMessage(Message msg) { super.handleMessage(msg); Log.i(TAG, "handleMessage"); switch (msg.what) { case 1: //RemoteViews的AIDL實(shí)現(xiàn) RemoteViews remoteViews = msg.getData().getParcelable("remoteViews"); if (remoteViews != null) { Log.i(TAG, "updateUI"); View view = remoteViews.apply(weakReference.get(), mLinearLayout); mLinearLayout.addView(view); } break; default: break; } } }; public void readyGo(View view){ Intent intent = new Intent(MainActivity.this, Temp2Activity.class); startActivity(intent); } }
Temp2Activity.java
package com.czhappy.remoteviewdemo.activity; import android.app.PendingIntent; import android.content.ComponentName; import android.content.Context; import android.content.Intent; import android.content.ServiceConnection; import android.os.Bundle; import android.os.IBinder; import android.os.RemoteException; import android.support.annotation.Nullable; import android.support.v7.app.AppCompatActivity; import android.util.Log; import android.view.View; import android.widget.RemoteViews; import com.czhappy.remoteviewdemo.IremoteViewsManager; import com.czhappy.remoteviewdemo.R; import com.czhappy.remoteviewdemo.service.RemoteViewsAIDLService; /** * Description: * User: chenzheng * Date: 2017/2/9 0009 * Time: 16:05 */ public class Temp2Activity extends AppCompatActivity { private String TAG = "Temp2Activity"; private IremoteViewsManager remoteViewsManager; private boolean isBind = false; private ServiceConnection remoteViewServiceConnection = new ServiceConnection() { @Override public void onServiceConnected(ComponentName name, IBinder service) { Log.i(TAG,"onServiceConnected"); remoteViewsManager = IremoteViewsManager.Stub.asInterface(service); } @Override public void onServiceDisconnected(ComponentName name) { //回收 remoteViewsManager = null; } }; @Override protected void onCreate(@Nullable Bundle savedInstanceState) { super.onCreate(savedInstanceState); this.setContentView(R.layout.activity_temp); } /** * 綁定服務(wù) */ public void bindService(View view) { Intent viewServiceIntent = new Intent(this,RemoteViewsAIDLService.class); isBind = bindService(viewServiceIntent,remoteViewServiceConnection, Context.BIND_AUTO_CREATE); } /** * 更新UI */ public void UpdateUI(View view){ RemoteViews remoteViews = new RemoteViews(Temp2Activity.this.getPackageName(),R.layout.button_layout); Intent intentClick = new Intent(Temp2Activity.this,FirstActivity.class); PendingIntent openFirstActivity = PendingIntent.getActivity(Temp2Activity.this,0,intentClick,0); remoteViews.setOnClickPendingIntent(R.id.firstButton,openFirstActivity); Intent secondClick = new Intent(Temp2Activity.this,SecondActivity.class); PendingIntent openSecondActivity = PendingIntent.getActivity(Temp2Activity.this,0,secondClick,0); remoteViews.setOnClickPendingIntent(R.id.secondButton,openSecondActivity); remoteViews.setCharSequence(R.id.secondButton,"setText","想改就改"); try { remoteViewsManager.addRemoteView(remoteViews); } catch (RemoteException e) { e.printStackTrace(); } } @Override protected void onDestroy() { super.onDestroy(); if(isBind){ unbindService(remoteViewServiceConnection); isBind = false; } } }
AndroidManifest.xml
<?xml version="1.0" encoding="utf-8"?> <manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.czhappy.remoteviewdemo"> <application android:allowBackup="true" android:icon="@mipmap/ic_launcher" android:label="@string/app_name" android:supportsRtl="true" android:theme="@style/AppTheme"> <activity android:name=".activity.MainActivity"> <intent-filter> <action android:name="android.intent.action.MAIN" /> <category android:name="android.intent.category.LAUNCHER" /> </intent-filter> </activity> <activity android:name=".activity.FirstActivity" /> <activity android:name=".activity.SecondActivity" /> <activity android:name=".activity.Temp2Activity" android:process=":remote2" /> <service android:name="com.czhappy.remoteviewdemo.service.RemoteViewsAIDLService" /> </application> </manifest>
四、總結(jié)
RemoteViews就是為跨進(jìn)程更新UI而生的,內(nèi)部封裝了很多方法用來實(shí)現(xiàn)跨進(jìn)程更新UI。但這并不代表RemoteViews是就是萬能的了,它也有不足之處,目前支持的布局和View有限
layout:
FrameLayout LinearLayout RelativeLayout GridLayout
View:
AnalogClock button Chronometer ImageButton ImageView ProgressBar TextView ViewFlipper ListView GridView StackView AdapterViewFlipper ViewStub
不支持自定義View 所以具體使用RemoteViews還是aidl或者BroadCastReceiver還得看實(shí)際的需求。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- Android應(yīng)用程序四大組件之使用AIDL如何實(shí)現(xiàn)跨進(jìn)程調(diào)用Service
- Android 跨進(jìn)程模擬按鍵(KeyEvent )實(shí)例詳解
- Android AIDL實(shí)現(xiàn)兩個APP間的跨進(jìn)程通信實(shí)例
- Android編程實(shí)現(xiàn)AIDL(跨進(jìn)程通信)的方法詳解
- Android IPC機(jī)制利用Messenger實(shí)現(xiàn)跨進(jìn)程通信
- 詳解Android跨進(jìn)程IPC通信AIDL機(jī)制原理
- Android 跨進(jìn)程SharedPreferences異常詳解
- Android跨進(jìn)程拋異常的原理的實(shí)現(xiàn)
- Android 跨進(jìn)程通Messenger(簡單易懂)
- Android實(shí)現(xiàn)跨進(jìn)程接口回掉的方法
相關(guān)文章
Android編程設(shè)計(jì)模式之觀察者模式實(shí)例詳解
這篇文章主要介紹了Android編程設(shè)計(jì)模式之觀察者模式,結(jié)合實(shí)例形式詳細(xì)分析了Android觀察者模式的概念、功能、實(shí)現(xiàn)方法及相關(guān)使用注意事項(xiàng),需要的朋友可以參考下2017-12-12

Flutter交互并使用小工具管理其狀態(tài)widget的state詳解

詳解Android Studio中Git的配置及協(xié)同開發(fā)

詳解Android 7.0 Settings 加載選項(xiàng)