分析一下java层中的一个函数在C++层MediaPlayer中的过程,我们用setDataSource来看C/S模式的过程
对应查看MediaPlayerService中的create
函数
在图中有一个IPCThreadState
。在Android中ProcessState是客户端和服务端公共的部分,作为Binder通信的基础。ProcessState
是一个singleton
类,每个进程只有一个对象,这个对象负责打开Binder驱动,建立线程池,让其进程里面的所有线程都能通过Binder通信。与之相关的是IPCThreadState,每个线程都有一个IPCThreadState实例
登记在Linux线程的上下文附属数据中,主要负责Binder的读取、写入和请求处理。IPCThreadState在构造的时候获取进程的ProcessState并记录在自己的成员变量mProcess
中,通过mProcess可以获得Binder的句柄。IPCThreadState通过IPCThreadState::transact
把data及handle等填充进binder_transaction_data,在两个进程间通信
在MediaPlayerService.h中看看这个Client是什么
以上省略了部分代码,我们可以看到以上代码对应java层的MediaPlayer相关方法,在之前的具体类依赖关系图中我们可以整体理解这个Client属于什么角色及位置,它继承了BnMediaPlayer并包含IMediaPlayer相关接口
Client类的继承关系为Client->BnMediaPlayer->IMediaPlayer
。分析上面代码可看出,create函数构造了一个Client对象,并将该对象添加到MediaPlayerService类的全局列表mClients
中,这是一个SortedVector,紧接着执行player->setDataSource(url,headers)
,即Clients::setDataSource
,因此在setDataSource中有如下语句
等价于sp
,可以直接认为player==Client
Client是MediaPlayerService
内部一个类,从上方的代码可知MediaPlayerService运行在服务器端,故Client也运行在服务器端
Client在MediaPlayerService.h
中,接下来看一下MediaPlayerService中的实现,实现过程中调用过MediaPlayerService类的一些函数,同样会用到setDaataSource
然后重新看看MediaPlayer中头文件(frameworks/av/include/media/mediaplayer.h
)定义的函数声明,方便对比Client中的函数(部分展示)
这里的函数和Client的函数是一一对应的,两者通过Client的代理类联系在了一起
回看MediaPlayer::setDataSource
函数
查看其中的attachNewPlayer
setDataSource会调到attachNewPlayer函数,这个函数最终会调用服务器端Client对应的函数。
1.包结构:IMediaPlayer和IMediaPlayerClient.h都在frameworks/av/media/libmedia
包中,而mediapalyer.h在av/include/media
包中
2.功能职责不一样
IMediaPlayer.h
在IMediaPlayer.h中定义的基本上都是虚函数,我们知道虚函数在C++中用于实现多态性,多态性是将接口与具体实现代码进行了分离,用形象的语言解释就是以共同的方法实现,但因个体差异而采用不同策略。所以它的功能是实现MediaPlayer功能的接口,看到onTransact
函数,自然联想Binder通信,把底层的Parcel指针类型数据向上层的另一个进程传递
IMediaPlayerClient.h
上方代码总结:在内部定义一个BpMediaPlayerClient类(也就是Client的父类),然后它有一个onTransact函数。一般onXXX都是被动回调过来的,不是由自己控制的,如Activity中的onCreate等函数,这里也是一样,onTransact作为Binder通信中的回调函数,IMediaPlayerClient.h的功能是描述一个MediaPlayer客户端的接口。
mediaplayer.h
的功能是对外(JNI层)的接口类,它最主要的是定义了一个MediaPlayer类(C++层),我们在android_media_MediaPlayer.cpp中就引入了media/mediaplayer.h
IMediaPlayer.h
则是一个实现MediaPlayer(C++层)功能的接口
IMediaPlayerClient.h
的功能是描述一个MediaPlayer客户端(暂且理解为前面说的Client)的接口