阿里P7Android高级架构进阶视频免费学习请点击:https://space.bilibili.com/474380680
本篇文章将先从以下三个内容来介绍通信方案:

  • [Activity与Fragment之间的通信交互]
  • [ Service和Activity的相互通信]
  • [Intent传递数据大小限制]

一、Activity与Fragment之间的通信交互

1.1、APP构成

MainActivity+底部导航栏;

1.2、通信目的

MainActivity中成员方法test(),fragment中经test()方法判断后,方可执行下一步操作,如执行界面跳转;
例:

  1. public class MainActivity extends Activity{
  2. public boolean mBaiDuSDK_is_Ok = false;
  3. public boolean ismBaiDuSDK_is_Ok() {
  4. return mBaiDuSDK_is_Ok; }
  5. public void setmBaiDuSDK_is_Ok(boolean mBaiDuSDK_is_Ok) {
  6. this.mBaiDuSDK_is_Ok = mBaiDuSDK_is_Ok; }
  7. }

如上述示例代码,在fragment中需要对mBaiDuSDK_is_Ok进行值判断,执行界面跳转;

1.3、解决方法

  1. public class Test1Fragment extends Fragment implements OnClickListener {
  2. private Activity mActivity;
  3. @Override
  4. public View onCreateView(LayoutInflater inflater, ViewGroup container,
  5. Bundle savedInstanceState) {
  6. mActivity = getActivity();//在界面创建时,定义父Activity
  7. view = inflater.inflate(R.layout.fragment_test1, container, false);
  8. return view; }
  9. @Override public void onViewCreated(View view, Bundle savedInstanceState) {
  10. // fragment_test1 布局中的一个测试按钮
  11. Button test_button= (Button) view.findViewById(R.id. test_button);
  12. test_button setOnClickListener(this);//设置按钮监听事件
  13. super.onViewCreated(view, savedInstanceState);
  14. }
  15. @Override public void onClick(View view) {
  16. // TODO Auto-generated method stub switch (view.getId()) {
  17. //测试按钮监听事件
  18. case R.id.test_button:
  19. // mActivity即为代表的父Activity
  20. Intent intent = new Intent(mActivity, SecondActivity.class);
  21. Log.i("BAIDUSDK", "验证:" + ((MainActivity) mActivity).ismBaiDuSDK_is_Ok());
  22. //最为关键的一步,fragment调用父activity的成员方法,进行通信
  23. if (((MainActivity) mActivity).ismBaiDuSDK_is_Ok()) {
  24. startActivity(intent);
  25. } else {
  26. Log.e("TEST","请确认mBaiDuSDK_is_Ok是否正常!");
  27. }
  28. break;
  29. }
  30. }
  31. }

二、Service和Activity的相互通信

2.1第一种方式:通过MyBinder方式调用Service方法

步骤

  • 继承Binder 定义中间人对象

BanZhengService

  1. public class BanZhengService extends Service {
  2. //把我定义的中间人对象返回
  3. @Override
  4. public IBinder onBind(Intent intent) {
  5. return new MyBinder();
  6. }
  7. //办证的方法
  8. public void banZheng(int money){
  9. if (money>1000) {
  10. Toast.makeText(getApplicationContext(), "我是领导 把证给你办了", 1).show();
  11. }else {
  12. Toast.makeText(getApplicationContext(), "这点钱 还想办事....", 1).show();
  13. }
  14. }
  15. //[1]定义中间人对象(IBinder)
  16. public class MyBinder extends Binder{
  17. public void callBanZheng(int money){
  18. //调用办证的方法
  19. banZheng(money);
  20. }}}

  • 重写ServiceConnection,onServiceConnected时调用中间人对象 绑定服务
    MainActivity
  1. public class MainActivity extends Activity {
  2. private MyConn conn;
  3. private MyBinder myBinder;//我定义的中间人对象
  4. @Override
  5. protected void onCreate(Bundle savedInstanceState) {
  6. super.onCreate(savedInstanceState);
  7. setContentView(R.layout.activity_main);
  8. Intent intent = new Intent(this,BanZhengService.class);
  9. //连接服务
  10. conn = new MyConn();
  11. bindService(intent, conn, BIND_AUTO_CREATE);
  12. }
  13. //点击按钮调用服务里面办证的方法
  14. public void click(View v) {
  15. myBinder.callBanZheng(10000000);
  16. }
  17. //监视服务的状态
  18. private class MyConn implements ServiceConnection{
  19. //当服务连接成功调用
  20. @Override
  21. public void onServiceConnected(ComponentName name, IBinder service) {
  22. //获取中间人对象
  23. myBinder = (MyBinder) service;
  24. }
  25. //失去连接
  26. @Override
  27. public void onServiceDisconnected(ComponentName name) {
  28. }}
  29. @Override
  30. protected void onDestroy() {
  31. //当activity 销毁的时候 解绑服务
  32. unbindService(conn);
  33. super.onDestroy();
  34. }}

2.2 第二种方式:通过接口Iservice调用Service方法

使用借口调用service和直接调用其实本质都是一样的,只不过多了借口一个步骤,

即实现步骤

  • 1.继承Binder 定义中间人对象
  • 2.定义接口
  1. public interface Iservice {
  2. //把领导想暴露的方法都定义在接口里
  3. public void callBanZheng(int money);
  4. // public void callPlayMaJiang();

  • 3.重写ServiceConnection,onServiceConnected时调用中间人对象,强转为接口(myBinder = (Iservice) service;) 绑定服务

这里就写一下不同的地方,其他都和上面的第一种一样
MainActivity


  1. //监视服务的状态
  2. private class MyConn implements ServiceConnection{
  3. //当服务连接成功调用
  4. @Override
  5. public void onServiceConnected(ComponentName name, IBinder service) {
  6. //获取中间人对象
  7. myBinder = (Iservice) service;
  8. }
  9. //失去连接
  10. @Override
  11. public void onServiceDisconnected(ComponentName name) {
  12. }
  13. }
  14. @Override
  15. protected void onDestroy() {
  16. //当activity 销毁的时候 解绑服务
  17. unbindService(conn);
  18. super.onDestroy();
  19. }
  20. }

三、 Intent传递数据大小限制

3.1前言

在sendBroadcast,startActivity时,我们会用到Intent。
Intent可以携带一些数据,比如基本类型数据int、Boolean,或是String,或是序列化对象,Parcelable与Serializable。
Intent传递数据时,如果数据太大,可能会出现异常。比如App闪退,或是Intent发送不成功,logcat报错等等。
这就牵涉到一个问题:Intent 传递数据大小限制。
Intent到底能够携带多少数据呢?
使用Intent传送数据时,可能会出现异常
在Intent中传入一个Parcelable对象;例如传入一个bitmap对象。

  1. Bitmap b1 = Bitmap.createScaledBitmap(srcBmp, dstWid, dstHeight, false);
  2. Intent intent = new Intent(MSG_INTENT);
  3. intent.putExtra(K_PIC, b1);

选择bitmap的原因是,Bitmap实现了Parcelable接口,并且可以通过getByteCount()得知所占内存大小。
sendBroadcast时,报出如下信息:

  1. V/ActivityManager: Broadcast: Intent { act=intent_bi flg=0x10 (has extras) } ordered=false userid=0 callerApp=ProcessRecord{27aeaaf5 31217:com.rustfisher.basic4/u0a113}
  2. E/JavaBinder: !!! FAILED BINDER TRANSACTION !!!
  3. W/BroadcastQueue: Failure sending broadcast Intent { act=intent_bi flg=0x10 (has extras) }
  4. android.os.TransactionTooLargeException
  5. at android.os.BinderProxy.transactNative(Native Method)
  6. at android.os.BinderProxy.transact(Binder.java:504)
  7. atandroid.app.ApplicationThreadProxy.scheduleRegisteredReceiver(ApplicationThreadNative.java:1170) atcom.android.server.am.BroadcastQueue.performReceiveLocked(BroadcastQueue.java:576 atcom.android.server.am.BroadcastQueue.deliverToRegisteredReceiverLocked(BroadcastQueue.java:848) atcom.android.server.am.BroadcastQueue.processNextBroadcast(BroadcastQueue.java:91 atcom.android.server.am.BroadcastQueue$BroadcastHandler.handleMessage(BroadcastQueue.java:254)
  8. at android.os.Handler.dispatchMessage(Handler.java:111)
  9. at android.os.Looper.loop(Looper.java:194)
  10. at android.os.HandlerThread.run(HandlerThread.java:61)
  11. at com.android.server.ServiceThread.run(ServiceThread.java:46)

查看异常类TransactionTooLargeException,它继承了RemoteException

  1. package android.os; public class TransactionTooLargeException extends RemoteException { public TransactionTooLargeException() {
  2. super();
  3. }
  4. public TransactionTooLargeException(String msg) {
  5. super(msg);
  6. }
  7. }

追踪到Binder,它的transactNative方法会报出RemoteException

  1. public native boolean transactNative(int code, Parcel data, Parcel reply,
  2. int flags) throws RemoteException;

抛出异常与Binder有关。

Intent携带信息的大小受Binder限制

Intent携带信息的大小其实是受Binder限制。本文标题也可以改为“Binder传递数据大小限制”。

数据以Parcel对象的形式存放在Binder传递缓存中。

如果数据或返回值比传递buffer大,则此次传递调用失败并抛出TransactionTooLargeException异常。

Binder传递缓存有一个限定大小,通常是1Mb。但同一个进程中所有的传输共享缓存空间。

多个地方在进行传输时,即时它们各自传输的数据不超出大小限制,TransactionTooLargeException异常也可能会被抛出。

在使用Intent传递数据时,1Mb并不是安全上限。因为Binder中可能正在处理其它的传输工作。

不同的机型和系统版本,这个上限值也可能会不同。

在其它地方,例如onSaveInstanceState(@NonNull Bundle outState),也可能会遇到与Binder有关的类似问题

为什么Binder要限制传输数据的大小

个人推测,作为一种IPC的方式,Binder并不是为传输大量数据而设计。

传输大量数据,可以考虑URL之类的方法。
阿里P7Android高级架构进阶视频免费学习请点击:https://space.bilibili.com/474380680
参考
https://www.jb51.net/article/159479.htm
https://www.jianshu.com/p/4b0cbe068555
https://www.jb51.net/article/120494.htm

Android组件内核之组件间通信方案(四)上篇的更多相关文章

  1. Android组件内核之间组件间通信方案(四)下篇

    阿里P7Android高级架构进阶视频免费学习请点击:https://space.bilibili.com/474380680本篇文章将继续从以下两个内容来介绍通信方案: [ViewModel 与 V ...

  2. android 中的 Handler 线程间通信

    一. 在MainActivity中为什么只是类似的写一行如下代码就可以使用handler了呢? Handler handler = new Handler() { @Override public v ...

  3. Linux-进程间通信(四): 域套接字

    1. 域套接字: (1) 只能用于同一设备上不同进程之间的通信: (2) 效率高于网络套接字.域套接字仅仅是复制数据,并不走协议栈: (3) 可靠,全双工: 2. 域套接字地址结构: struct s ...

  4. Android中线程间通信原理分析:Looper,MessageQueue,Handler

    自问自答的两个问题 在我们去讨论Handler,Looper,MessageQueue的关系之前,我们需要先问两个问题: 1.这一套东西搞出来是为了解决什么问题呢? 2.如果让我们来解决这个问题该怎么 ...

  5. CAP-微服务间通信实践

    微服务间通信常见的两种方式 由于微服务架构慢慢被更多人使用后,迎面而来的问题是如何做好微服务间通信的方案.我们先分析下目前最常用的两种服务间通信方案. gRPC(rpc远程调用) gRPC-微服务间通 ...

  6. Cisco基础(二):三层交换vlan间通信、多交换机vlan间通信、三层交换配置路由、RIP动态路由配置、三层交换配置RIP动态路由

    一.三层交换vlan间通信 目标: VLAN实现了广播域的隔离,同时也将VLAN间的通信隔离了.三层交换技术使得VLAN间可以通信. 通过三层交换实现VLAN间通信 方案: 为了解决了传统路由器低速. ...

  7. gRPC-微服务间通信实践

    微服务间通信常见的两种方式 由于微服务架构慢慢被更多人使用后,迎面而来的问题是如何做好微服务间通信的方案.我们先分析下目前最常用的两种服务间通信方案. gRPC(rpc远程调用) 场景:A服务主动发起 ...

  8. Android 组件间通信--事件驱动

    在android中,组件间通信常用的方式: 1.使用广播机制:在主页面中监听特定的广播事件,进行业务逻辑的操作,其他页面只需要根据需求发送广播即可 例如:常用app结构中,左边通常为菜单栏,点击菜单栏 ...

  9. React独立组件间通信联动

    React是现在主流的高效的前端框架,其官方文档 http://reactjs.cn/react/docs/getting-started.html 在介绍组件间通信时只给出了父子组件间通信的方法,而 ...

随机推荐

  1. Learning OSG programing---osgClip

    OSG Clip例程剖析 首先是创建剪切节点的函数代码: osg::ref_ptr<osg::Node> decorate_with_clip_node(const osg::ref_pt ...

  2. 利用Graphziv帮助理解复杂的类层次关系

    最近在学习osg三维视景仿真平台,学习的过程中涉及到许多的类与类之间的继承和包含关系.在复杂点的例子中,许多的类和节点组合在一起,很容易让人迷失方向.在编译源代码的时候,无意间发现了Graphviz这 ...

  3. 深入理解javascript原型和闭包(4)——隐式原型 (转载)

    深入理解javascript原型和闭包(4)——隐式原型   注意:本文不是javascript基础教程,如果你没有接触过原型的基本知识,应该先去了解一下,推荐看<javascript高级程序设 ...

  4. Python 学习笔记12 函数模块

    函数的优点之一,使用它们可将代码块与主程序分离.通过给函数指定描述性的名称.可以让主程序非常好理解.但是如果将过多的函数和主程序放置在一起,会让文件显得非常凌乱.太多的代码混杂在一起,不方便管理.我们 ...

  5. 绿盟扫出来个http host 漏洞

    这个漏洞搞了大半天,想过从后台拦截,也想过从前台拦截,都是无从下手!网上也找了很多资料,有点乱,后来自己结合网上的办法,搞出如下解决办法:在tomcat server.xml里配置host 因为外网是 ...

  6. android ListView列表显示数据

    item.xml <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:and ...

  7. 【问题解决方案】git/github本地和远程仓库的重命名

    参考: CSDN博文:在Github上重命名仓库 背景: 偶然终于看到一条规范里写着: "通常(注意是通常,尤其是 Web 相关的项目) repo 的命名用小写英文,多个字母之间用连字符(比 ...

  8. 修改bug 提交出错:操作失败: 无法更改关系,因为一个或多个外键属性不可以为 null

    提交出错:操作失败: 无法更改关系,因为一个或多个外键属性不可以为 null.对关系作出更改后,会将相关的外键属性设置为 null 值.如果外键不支持 null 值,则必须定义新的关系,必须向外键属性 ...

  9. 字符串String的使用方法

    var ddd = "举头望明月,低头思故乡" document.writeln(ddd.split(''));//选择字符串中的一个标识符,将字符串分割成数组; var slic ...

  10. services - Internet 网络服务列表

    DESCRIPTION(描述) services 是一个普通的 ASCII 码文件, 它在 internet 服务的友好原文名以及这些服务预先分配的端口和协议类型之间提供了映射. 每个联网程序必须查找 ...