亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

詳解android webView獨立進程通訊方式

發布時間:2020-10-06 12:27:39 來源:腳本之家 閱讀:319 作者:區區一只yamada 欄目:移動開發

為什么需要將webView放在獨立進程

  • webView 加載網頁的時候可能占用大量內存,導致應用程序OOM。
  • webView 在訪問結束的時候可以直接殺死該進程,防止內存泄漏。
  • webView 在崩潰的時候不影響主進程。

webView獨立進程需要注意什么

  • 由于進程之間內存是獨立的,所以導致了Appcation, 靜態類需要在新的進程重新創建。
  • 內存中的數據不共享,需要跨進程通訊。

如何聲明一個獨立進程

在默認情況下,同一應用的所有組件都在相同的進程中運行。
在Manifest中可以設置各組件 (<activity>、<service>、<receiver>、<provider>)的 android:process 屬性來指定相應的進程。

跨進程的方式

在android當中提供了2種方式實現。

一種是Messenger, 另一種是Aidl.

  • Messenger:實現相對簡單,將所有請求放到消息隊列中,不適合做并發處理,在大多數的場景用Messenger就可以實現了。
  • AIDL: 適合并發操作。直接方法調用,結構更清晰。

Messenger

由于Messenger是采用消息隊列的方式實現,所有接受和發送的時候都需要Handler協助。

服務端

public class MessengerService extends Service {
  
  public static final int GET_DATA = 1;
  public static final int SET_DATA = 2;
  
  Messenger messenger = new Messenger(new ServiceHandler());
  Messenger replyMessenger; //向客服端返回信息
  public MessengerService() {
  }
  
  @Override
  public IBinder onBind(Intent intent) {
    return messenger.getBinder();
  }
  
  
  class ServiceHandler extends Handler {
    @Override
    public void handleMessage(Message msg) {
      replyMessenger = msg.replyTo;
      switch (msg.what) {
        case GET_DATA:
          //客服端向服務端請求數據
          if (replyMessenger != null) {
            Bundle bundle = new Bundle();
            bundle.putString("str", CustomData.getInstance().getData());
            Message message = Message.obtain(null, 1);
            message.setData(bundle);
            try {
              replyMessenger.send(message);
            } catch (RemoteException e) {
              e.printStackTrace();
            }
          }
          break;
        case SET_DATA:
          //客服端向服務端請求更新數據
          CustomData.getInstance().setData(msg.getData().getString("str"));
          break;
      }
    }
  }
}

客服端:

public class MessengerClientActivity extends AppCompatActivity {
  
  private WebView mWebView;
  private Button mGetDatBtn;
  private Button mSetDatBtn;
  
  public static void startThis(Context context, String url) {
    Intent intent = new Intent(context, MessengerClientActivity.class);
    intent.putExtra("url", url);
    context.startActivity(intent);
  }
  
  @Override
  protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_messenger_client);
    mWebView = (WebView) findViewById(R.id.webview);
    mGetDatBtn = (Button) findViewById(R.id.get_data_btn);
    mSetDatBtn = (Button) findViewById(R.id.set_data_btn);
        
    WebSettings webSettings = mWebView.getSettings();
    webSettings.setJavaScriptEnabled(true);
    webSettings.setJavaScriptCanOpenWindowsAutomatically(true);
    webSettings.setSupportZoom(false);
    webSettings.setBuiltInZoomControls(false);
    webSettings.setAllowFileAccess(true);
    webSettings.setDatabaseEnabled(true);
    webSettings.setDomStorageEnabled(true);
    webSettings.setGeolocationEnabled(true);
    webSettings.setAppCacheEnabled(true);
    webSettings.setAppCachePath(getApplicationContext().getCacheDir().getPath());
    webSettings.setDefaultTextEncodingName("UTF-8");
    //屏幕自適應
    webSettings.setUseWideViewPort(true);
    webSettings.setLoadWithOverviewMode(true);
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
      webSettings.setCacheMode(WebSettings.LOAD_CACHE_ELSE_NETWORK);
    } else {
      webSettings.setCacheMode(WebSettings.LOAD_DEFAULT);
    }
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.HONEYCOMB) {
      webSettings.setDisplayZoomControls(false);
    }
    if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.KITKAT) {
      webSettings.setLoadsImagesAutomatically(true);
    } else {
      webSettings.setLoadsImagesAutomatically(false);
    }
    
    mWebView.setScrollBarStyle(WebView.SCROLLBARS_INSIDE_OVERLAY);
    mWebView.setHorizontalScrollBarEnabled(false);
    mWebView.setHorizontalFadingEdgeEnabled(false);
    mWebView.setVerticalFadingEdgeEnabled(false);
    
    String url = "http://www.jianshu.com/";
    mWebView.loadUrl(url);

    mGetDatBtn.setOnClickListener(new View.OnClickListener() {
      @Override
      public void onClick(View v) {
        getData();
      }
    });
    
    mSetDatBtn.setOnClickListener(new View.OnClickListener() {
      @Override
      public void onClick(View v) {
        setData();
      }
    });
  }
  
  Messenger messenger;
  Messenger messengerReply = new Messenger(new Handler() {
    @Override
    public void handleMessage(Message msg) {
      switch (msg.what) {
        case MessengerService.GET_DATA:
          mGetDatBtn.setText("" + msg.getData().get("str"));
          break;
      }
    }
  });
  boolean mBound;
  ServiceConnection serviceConnection = new ServiceConnection() {
    @Override
    public void onServiceConnected(ComponentName name, IBinder service) {
      messenger = new Messenger(service);
      mBound = true;
    }
  
    @Override
    public void onServiceDisconnected(ComponentName name) {
      messenger = null;
      mBound = false;
    }
    
  };
  
  private void getData() {
    if (!mBound) return;
    Message message = Message.obtain(null, MessengerService.GET_DATA, 0,0);
    //用于服務端應答
    message.replyTo = messengerReply;
    sendMessage(message);
  }
  
  private void setData() {
    if (!mBound) return;
    Message message = Message.obtain(null, MessengerService.SET_DATA, 0,0);
    sendMessage(message);
  }
  
  private void sendMessage(Message message) {
    try {
      messenger.send(message);
    } catch (RemoteException e) {
      e.printStackTrace();
    }
  }
  
  
  @Override
  protected void onStart() {
    super.onStart();
    // Bind to the service
    bindService(new Intent(this, TestWebService.class), serviceConnection,
        Context.BIND_AUTO_CREATE);
  }
  
  @Override
  protected void onStop() {
    super.onStop();
    // Unbind from the service
    if (mBound) {
      unbindService(serviceConnection);
      mBound = false;
    }
  }
  
  private void destroyWebView(WebView webView) {
    if (webView == null)
      return;
    webView.stopLoading();
    ViewParent viewParent = webView.getParent();
    if (viewParent != null && viewParent instanceof ViewGroup)
      ((ViewGroup) viewParent).removeView(webView);
    webView.removeAllViews();
    webView.destroy();
    webView = null;
  }
  
  @Override
  protected void onDestroy() {
    destroyWebView(mWebView);
    super.onDestroy();
  }
}

AIDL

第一步:創建.aidl文件

  • aidl默認支持以下的類型:
  • Java 編程語言中的所有原語類型(如 int、long、char、boolean 等等)
  • String
  • CharSequence
  • List
  • Map
  • 如果需要導入自己的類型需要加入一個 import 語句(注意:導入的類需要實現Parcelabel接口)

aidl文件:

interface IAidlProcess {

  //默認支持原語類型(int、long、char等等)、String、CharSequence、List、Map
  //自定義類型需要導入 import eebochina.com.testtechniques.testwebview.XXXClass
  //自定義類型傳輸一定需要是序列化對象
  String getCustomData();

  void setCustomData(String str);
}

服務端

public class AidlService extends Service {
  public AidlService() {
  }
  ITestProcess.Stub mBinder = new ITestProcess.Stub() {
    @Override
    public String getCustomData() throws RemoteException {
      return CustomData.getInstance().getData();
    }
  
    @Override
    public void setCustomData(String str) throws RemoteException {
      CustomData.getInstance().setData(str);
    }
  };
  @Override
  public IBinder onBind(Intent intent) {
    return mBinder;
  }
}

客服端獲取綁定接口

  AidlService mAidlService;
  private ServiceConnection serviceConnection = new ServiceConnection() {
    @Override
    public void onServiceConnected(ComponentName name, IBinder service) {
      mAidlService = IAidlProcess.Stub.asInterface(service);
      mBound = true;
    }
    
    @Override
    public void onServiceDisconnected(ComponentName name) {
      mBound = false;
      mAidlService = null;
    }
  };

在獲取了綁定接口后就可以直接和服務端通訊了。

2種通訊方式都簡單的介紹了下,后面的實際應用還需要根據不同的業務進行調整。

由于aidl是方法直接調用的,從代碼擴展和閱讀來說比messenger要強很多。

如果有寫的不好和不對的地方,希望大家可以及時指出來。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持億速云。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

宁河县| 晋江市| 湘潭县| 裕民县| 重庆市| 阿瓦提县| 读书| 桂平市| 本溪| 乐陵市| 军事| 龙游县| 萨嘎县| 庄河市| 祁连县| 孟津县| 溆浦县| 通海县| 鲁甸县| 长顺县| 香格里拉县| 都安| 河北区| 交口县| 铜山县| 喀喇沁旗| 吉安县| 顺昌县| 隆德县| 溆浦县| 鹤壁市| 藁城市| 宜宾市| 托克托县| 黑水县| 宿迁市| 博湖县| 九江市| 雷波县| 南宫市| 连云港市|