zoukankan      html  css  js  c++  java
  • Android进程间通讯

    最近研究了一下Android进程间通讯,原来只是会用,但是只是会用是不行滴,就来研究一下。

    刚开始看的时候,我的头是这么大,看了一夜的时候,头就变成这样了,,吓得宝宝赶紧上床休息了,、

    先喝喝茶讲个故事再来说这个通讯。

    (写完之后,看到这个终于承认了自己写作的能力很烂,但是好歹也是自己一个一个敲上去的,不能白白辛苦啊,嘿嘿)

    -----------------------------------------------------------------------------------------------------------------

      话说我有一哥们,最近他前女友来找他复合,但是他现在有女友,所以很是烦恼,整天吃不好睡不着,所以他打算问问上帝有什么好主意,可是为了安全起见,他又不能直接去找上帝,总不能一天死个5,6次吧,人生是没有bug的。那怎么办呢,幸好这世上还有一种生物叫神父,然后他就去找神父。

    “father ,我最近很烦恼,你可以帮我问问上帝该怎么办吗”

    “oh my son 上帝是万能的,他可以帮我们解决任何问题,有什么就说吧”

    “我现在有女朋友了,可是我的前女友最近又来找我复合,两个都对我很好,我该怎么办呢。。。。。(此处省略N多字)”

    神父把上帝从睡梦中喊醒“上帝啊,这人有病啊,有两个女的都喜欢他,直接收了不就完了,他还在纠结该选那个,您看这个怎么办才好呢”

    “哦,这件事我知道,他不是有个好朋友叫杨XX吗,我已经施法让他的前女友喜欢上杨XX了,哈哈”

    “son,上帝已经施法让你的前女友喜欢上了别人,是你的一个好朋友杨XX,你看这样好不”

    。。。。。。。。。。

    情景转到篮球场,我正在一群少女的欢呼中挥洒魅力,突然被一个手臂一拉。

    “XX,你知道我最近很烦心的那个事吗?”

    我正因为被他从少女的崇拜中打搅不爽“不就是那个前女友吗,怎么了”

    “哈哈,我今天去找上帝,上帝已经帮我解决了”

    “哦,怎么解决的,看你高兴的”

    “上帝施法让我的前女友喜欢你了”

    我眼前一黑,一口心血喷出“上帝,卧槽你X“

    我的女友是跆拳道黑带三段。

    ---------------------------------------------------------------------------------------------------------------

    (我又看了一遍,这个故事的确有点扯。。。。)

    好了,忘掉吧,你可能会感到这个故事跟进程间通讯半毛钱关系没有,但我感觉还是挺好的。。

    Android进程间通讯,

      为什么不能直接跨进程进行通讯啊?

      • 因为,为了安全考虑。Android系统中,应用所在的内存是独立的,无法相互访问,各个应用的数据都在自己的内存中。

      那么如何跨进程进行通讯呢?

      • 要想跨进程,必须找一个公共的大家都能访问的地方,通过暗号来达到通讯的目的。

      Android中的跨进程通讯采用的是binder机制。

      那么什么是binder呢,这个工作原理是什么呢,

      Binder呢,其实就是一块内存,它在Linux层面属于一个驱动,但是这个驱动不是驱动硬件,而是驱动一块内存。不同的应用通过对一块公共的内存进行数据的读写,来达到通讯的目的。而且应用之间进行通讯,

    必须要有暗号(没有暗号就成了火锅,乱套了!),当两个应用持有相同的暗号(AIDL时),才可以进行通讯。

    知道了为什么跨,怎么跨,那么具体步骤怎么写呢?

      由于不是本文重点,请看官方开发文档,现在开发文档越来越详细,估计人家美国12岁的小学生都能随便开发一个简单的应用。唉 英语啊。

    知道了以上这些东西,可是binder是怎么工作的,却还不知道,接下来为大家介绍一下binder的工作原理。

      在Android系统中的binder机制中,分别由Client,Server,ServiceManager和Binder驱动程序组成,其中Client、Server和Service Manager运行在用户空间,Binder驱动程序运行内核空间。Binder驱动程序是核

    心组件, Service Manager是一个守护进程,用来管理Server,并向Client提供查询Server接口的功能,开发者只需要在用户空间实现自己的Client和Server即可。而Client和Server正是在Binder驱动和ServiceManager提供的基础设施上,进行通信的。

      binder工作原理:

    binder通信是一种client-server的通信结构,

        1.从表面上来看,是client通过获得一个server的代理接口,对server进行直接调用;

        2.实际上,代理接口中定义的方法与server中定义的方法是一一对应的;

        3.client调用某个代理接口中的方法时,代理接口的方法会将client传递的参数打包成为Parcel对象;

        4.代理接口将该Parcel发送给内核中的binder driver.

        5.server会读取binder driver中的请求数据,如果是发送给自己的,解包Parcel对象,处理并将结果返回;

     6.整个的调用过程是一个同步过程,在server处理的时候,client会阻塞

    直接看不好理解,还好有图。

      前面已经说过,Service Manager组件是用来管理Server并且向Client提供查询Server远程接口的功能,那么,Service Manager就必然要和Server以及Client进行通信了。然而,Service Manager、Client和

    Server三者分别是运行在独立的进程当中,这样它们之间的通信也属于进程间通信了,而且也是采用Binder机制进行进程间通信,因此,ServiceManager在充当Binder机制的守护进程的角色的同时,也在充当Server

    的角色,也就是说,它是一种特殊的Server。

       其实在平时开发中,系统服务就是通过Binder机制来和应用通讯的,接下来为大家详细解释一下系统服务和应用通讯流程,以帮助大家更容易深刻的理解binder机制。

      系统服务和应用的通信机制

     

    系统服务:

      1.是一个Binder类的子类,一旦创建后,就开启一个线程死循环用来检测某段内存是否有数据写入。

      2.它自身创建时,创建一个xxxRemote远程对象,存放到Binder驱动中,xxxRemote远程对象可以和xxxService系统服务通信

    应用端:

      通过context. getSystemService获取xxxServiceProxy对象,该对象内部引用了xxxRemote对象, xxxServiceProxy和xxxService具有相同的API,我们调用xxxServiceProxy时,xxxServiceProxy就调用

    xxxRemote并等待xxxRemote返回。xxxRemote会往某段内存中写入数据,写完后就开始监视该内存区域,Binder驱动会把xxxRemote写入的数据拷贝到xxxService监视着的内存区域,当xxxService一旦发现有数

    据,就读取并进行处理,处理完毕后,就写入该区域,这时Binder驱动又会把该数据拷贝到xxxRemote监视的内存区域,当xxxRemote发现内存区域有数据读取该区域数据,并把内容返回给xxxServiceProxy。这样

    完成了一次进程间的通信。

      所以一个系统服务会产生两个Binder对象,一个是运行在系统中的系统服务本身,一个是存放到Binder驱动中的远程对象。所不同的是系统服务Binder对象会开启一个线程监听消息,远程对象不会,它是运行在

    调用者的线程中。

      客户端也可以不使用系统服务的远程Binder对象,而是自己创建一个Binder对象,通过Binder驱动和系统服务进行关联,这样的好处客户端可以随时通知系统服务,系统服务也可以随时通知客户端,而不是像上

    面所说的系统服务只能被动的等着客户端调用。

     

    那么binder内部是怎么操作的

      Binder对象都有各自的内存区域,当Binder1想要向Binder2发送数据时,就会把数据写入自己的内存区域,然后通知Binder驱动,Binder驱动会把数据拷贝到Binder2的内存区

    域,然后通知Binder2进行读取,Binder2读取完毕后,将把数据写入binder2的内存区域,然后通知Binder驱动,Binder驱动将会把数据拷贝到Binder1的内存区域中。这样就完成

    了一次通信。

      如果Binder1是系统服务,Binder2是系统服务的远程对象,这样任何应用程序在获取了Binder2的引用后,都可以和Binder1进行通信。但是缺点也很明显,只能由应用端请求

    系统服务,系统服务不能主动去联系应用端。WifiManagerService之类的就是采用这种方式。

       还有一种方式是Binder1是系统服务,Binder2是应用端创建的Binder对象,他们两者通过Binder驱动进行连接后,应用端可以主动调系统服务,系统服务也可以主动调用应用

    端。WindowManagerService就是采用的这种方式。

     我是看了很多资料才写出的这篇博客,大家有兴趣可以去看看http://blog.csdn.net/luoshengyang/article/details/6618363#comments的博客,的确大牛。

  • 相关阅读:
    ecplise自动提示失效,使用补全自动提示快捷键(Alt+/),但只显示“No Default Proposals”
    maven构建ssh工程
    pom.xml中坐标的组成
    依赖传递的规则
    maven中导入包版本冲突的解决
    maven工程的拆分与聚合
    maven的生命周期
    maven的常用命令
    在pom.xml中引入jar包坐标的依赖范围
    【stl的神奇操作】用集合搞定区间相交
  • 原文地址:https://www.cnblogs.com/makaruila/p/4869944.html
Copyright © 2011-2022 走看看