1,概述
Android系统中的各应用程序都运行在各自的进程中,进程之间通常是无法直接交换数据的。
Android提供了跨进程调用Service的功能,称为AIDL(android interface define language)Android接口定义语言。
ADIL相当与两个进程通信的协议,通过这个协议对进程间的通信进行了规范。按照该规范编写代码即可实现进程间的通信。
2,AIDL 接口文件
跨进程调用服务中需要定义接口文件,扩展名为.aidl
1、在项目的src文件夹下定义一个AIDL接口文件。
2、AIDL接口的内部语法与Java很相似,后面会演示
3、.aidl接口文件创建后,Android系统会自动生成主文件名相同的.java接口文件。该文件在项目的gen文件夹下,该文件不能修改,是Android自动生成的。
3,具体操作
因为是跨进程的通信协议,需要创建两个项目,一个项目是后台程序,一个是启动该后台服务的客户端,通过启动这两个项目来演示2个进程间的通信。
步骤1:创建Service项目,将该项目中的Activity类删除,把AndroidManifest.xml中的activity配置删除。
步骤2:创建AIDL接口文件:
a,创建一个包,在该包下创建java类,IMyService.java
package com.example.aidl; interface IMyService { void play(); void pause(); }
b,将IMyService.java改名为IMyService.aidl。(需要到项目代码路径中去修改)
注意:
1.aidl文件中不能出现访问修饰符,如public
2.aidl文件(包括所在包)在两个项目中要完全一致
c,修改后刷新项目,得到如下图情况
其中,IMyService.java是Android根据IMyService.aidl文件自动生成的接口,打开该文件查看源代码如下:
说明:
标注1- Stub类是IMyService接口的内部抽象类,该类继承了Binder类。
标注2- 这是Stub类中的asInterface方法,该方法负责将service返回至client的对象转换为IMyService.Stub。。。。
标注3- 指向的两个方法是步骤2定义的IMyService接口中声明的两个方法。
步骤3:新建包结构,自定义AIDLService.java类,该类继承Service类。
package com.example.aidlService; import com.example.aidl.IMyService; import android.app.Service; import android.content.Intent; import android.os.Binder; import android.os.IBinder; import android.os.RemoteException; import android.util.Log; public class AIDLService extends Service { private static final String tag = "AIDLService"; private MyBinder mBinder; @Override public IBinder onBind(Intent intent) { Log.i(tag, "...service onBind()..."); return mBinder; } /* * 该类继承了IMyService.Stub类而不是extends Binder类。 * IMyService.Stub是Binder的子类。 * 进程内的Service定义MyBinder内部类是继承Binder类。 */ public class MyBinder extends IMyService.Stub { @Override public void play() throws RemoteException { AIDLService.this.play(); } @Override public void pause() throws RemoteException { AIDLService.this.pause(); } } @Override public void onCreate() { super.onCreate(); //创建MyBinder对象,并在onBind()方法中返回该对象 mBinder = new MyBinder(); Log.i(tag, "service onCreate()..."); } private void play() { Log.i(tag, "service 自定义 play()..."); } private void pause() { Log.i(tag, "service 自定义 pause()..."); } @Override public int onStartCommand(Intent intent, int flags, int startId) { Log.i(tag, "service onStartCommand..."); return super.onStartCommand(intent, flags, startId); } @Override public boolean onUnbind(Intent intent) { Log.i(tag, "service onUnbind()..."); return super.onUnbind(intent); } }
在AndroidManifet.xml中注册Service。
<service android:name="com.example.aidlService.AIDLService" > <intent-filter> <action android:name="com.example.service.aidl" /> </intent-filter> </service> <!-- 注意,这里指定action,提供给其他进程来访问 -->
步骤4:创建Client项目,界面如下:
步骤5:将Service项目中创建的IMyService.aidl接口连包一起复制到Client项目中。结果如下图:
步骤6:在Client项目中的MainActivity.java中调用Service项目中的服务。
public class MainActivity extends Activity implements OnClickListener { Button btnBind, btnPlay, btnPause; IMyService mBinder; // 接口的一个引用 boolean mIsBind = false; // 绑定值为true,未绑定制为false; private ServiceConnection mConn = new ServiceConnection() { @Override public void onServiceDisconnected(ComponentName name) { } @Override public void onServiceConnected(ComponentName name, IBinder service) { /* * 获得另一个进程中的Service传递过来的IBinder对象-service, * 用IMyService.Stub.asInterface方法转换该对象,这点与进程内的通信不同 */ mBinder = IMyService.Stub.asInterface(service); mIsBind = true; Log.i("MainActivity", "onServiceConnected...."); } }; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); btnBind = (Button) findViewById(R.id.btn_bind); btnPlay = (Button) findViewById(R.id.btn_play); btnPause = (Button) findViewById(R.id.btn_pause); btnBind.setOnClickListener(this); btnPlay.setOnClickListener(this); btnPause.setOnClickListener(this); } @Override public void onClick(View v) { Intent intent = new Intent(); int btn = v.getId(); switch (btn) { case R.id.btn_bind: intent.setAction(Constant.ACTION_AIDL); bindService(intent, mConn, BIND_AUTO_CREATE); break; case R.id.btn_play: if (mIsBind){ try { mBinder.play(); } catch (RemoteException e) { e.printStackTrace(); }//调用service中的play() } break; case R.id.btn_pause: if(mIsBind){ try { mBinder.pause(); } catch (Exception e) { e.printStackTrace(); } } break; } } }
结果:当依次点击按钮时,会调用Service项目中定义的服务。
常见问题:
1,跨进程绑定服务与本地绑定服务的对比
跨进程调用并绑定服务与绑定本地(同一应用程序内部的服务称为本地服务)服务有所不同。
1) 绑定本地服务是:本地的Service通过onBinder方法将装载数据的IBinder对象传递给客户端的ServiceConnection对象的ServiceConnected方法的第二个参数service。并用Service中的内部类(自定义类)进行转换,从而获得从服务中返回的对象,通过调用该对象中的方法或属性值达到与被绑定的服务交换数据和控制该服务的目的。
2)跨进程绑定服务,首先要定义一个扩展名是 aidl 的接口问,该接口文件中申明了帮绑定服务所提供的方法。这个接口文件要复制到客户端程序中。
在服务总定义内部类时,不是直接继承 Binder 类,而是继承 接口.Stub 类(因Stub类已继承了Binder)。
在客户端的onServiceConnected方法中用IMyService.Stub.asInterface()方法转换服务器端传递过来的对象。
mBinder = IMyService.Stub.asInterface(service);
2,使用AIDL无法实现进程间通讯
两个项目的aidl文件不在同一个包中。
3,在AndroidManifest.xml文件中配置AIDL服务,注意<action>标签中的android.name 的属性值就是客户端要引用该服务的ID,也就是Intent.setAction("xxxxxxx");(或者Intent构造方法的参数)。注意,这里在Client进程中不能通过new Intent(context,IMyService.class) 来启动。
相关推荐
"Android 进程间通信AIDL使用详解" Android 进程间通信(IPC)是一种复杂的技术,AIDL(Android Interface Definition Language)是 Android 系统中的一种进程间通信机制。AIDL 是一种描述语言,用于定义服务器和...
它允许一个应用暴露方法给其他应用调用,就像在同一个进程中调用一样。AIDL文件定义了方法签名,类型和参数,Android系统会根据AIDL自动生成对应的Java接口。 在"DrunkService"中,我们通常会定义一个AIDL文件,...
通过AIDL,开发者可以创建能够跨进程调用的方法,使得一个应用组件可以调用另一个应用组件中的方法,即便这两个组件处于不同的进程。 ### AIDL基础概念 1. **服务(Service)**:在Android中,服务通常运行在自己...
AB PLC例程代码项目案例 【备注】 1、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用!有问题请及时沟通交流。 2、适用人群:计算机相关专业(如计科、信息安全、数据科学与大数据技术、人工智能、通信、物联网、自动化、电子信息等)在校学生、专业老师或者企业员工下载使用。 3、用途:项目具有较高的学习借鉴价值,不仅适用于小白学习入门进阶。也可作为毕设项目、课程设计、大作业、初期项目立项演示等。 4、如果基础还行,或热爱钻研,亦可在此项目代码基础上进行修改添加,实现其他不同功能。 欢迎下载!欢迎交流学习!不清楚的可以私信问我!
kolesar_3cd_01_0716
latchman_01_0108
matlab程序代码项目案例 【备注】 1、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用!有问题请及时沟通交流。 2、适用人群:计算机相关专业(如计科、信息安全、数据科学与大数据技术、人工智能、通信、物联网、自动化、电子信息等)在校学生、专业老师或者企业员工下载使用。 3、用途:项目具有较高的学习借鉴价值,不仅适用于小白学习入门进阶。也可作为毕设项目、课程设计、大作业、初期项目立项演示等。 4、如果基础还行,或热爱钻研,亦可在此项目代码基础上进行修改添加,实现其他不同功能。 欢迎下载!欢迎交流学习!不清楚的可以私信问我!
pimpinella_3cd_01_0716
petrilla_01_0308
AB PLC例程代码项目案例 【备注】 1、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用!有问题请及时沟通交流。 2、适用人群:计算机相关专业(如计科、信息安全、数据科学与大数据技术、人工智能、通信、物联网、自动化、电子信息等)在校学生、专业老师或者企业员工下载使用。 3、用途:项目具有较高的学习借鉴价值,不仅适用于小白学习入门进阶。也可作为毕设项目、课程设计、大作业、初期项目立项演示等。 4、如果基础还行,或热爱钻研,亦可在此项目代码基础上进行修改添加,实现其他不同功能。 欢迎下载!欢迎交流学习!不清楚的可以私信问我!
内容概要:本文档由张卓老师讲解,重点探讨DeepSeek的技术革新及强化学习对未来AI发展的重要性。文章回顾了AI的历史与发展阶段,详细解析Transformer架构在AI上半场所起到的作用,深入介绍了MoE混合专家以及MLA低秩注意机制等技术特点如何帮助DeepSeek在AI中场建立优势,并探讨了当前强化学习的挑战和边界。文档不仅提及AlphaGo和小游戏等成功案例来说明强化学习的强大力量,还提出了关于未来人工通用智能(AGI)的展望,特别是如何利用强化学习提升现有LLMs的能力和性能。 适用人群:本资料适宜对深度学习感兴趣的研究人员、开发者以及想要深入了解人工智能最新进展的专业人士。 使用场景及目标:通过了解最新的AI技术和前沿概念,在实际工作中能够运用更先进的工具和技术解决问题。同时为那些寻求职业转型或者学术深造的人提供了宝贵的参考。 其他说明:文中提到了许多具体的例子和技术细节,如DeepSeek的技术特色、RL的理论背景等等,有助于加深读者对于现代AI系统的理解和认识。
有师傅小程序开源版v2.4.14 新增报价短信奉告 优化部分细节
AB PLC例程代码项目案例 【备注】 1、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用!有问题请及时沟通交流。 2、适用人群:计算机相关专业(如计科、信息安全、数据科学与大数据技术、人工智能、通信、物联网、自动化、电子信息等)在校学生、专业老师或者企业员工下载使用。 3、用途:项目具有较高的学习借鉴价值,不仅适用于小白学习入门进阶。也可作为毕设项目、课程设计、大作业、初期项目立项演示等。 4、如果基础还行,或热爱钻研,亦可在此项目代码基础上进行修改添加,实现其他不同功能。 欢迎下载!欢迎交流学习!不清楚的可以私信问我!
商城二级三级分销系统(小程序+后台含源码).zip
li_3ck_01b_0918
nicholl_3cd_01_0516
媒体关注度是一个衡量公众对某个事件、话题或个体关注程度的重要指标。它主要反映了新闻媒体、社交媒体、博客等对于某一事件、话题或个体的报道和讨论程度。 媒体监督的J-F系数(Janis-Fadner系数)是一种用于测量媒体关注度的指标,特别是用于评估媒体对企业、事件或话题的监督力度。J-F系数基于媒体报道的正面和负面内容来计算,从而为公众、研究者或企业提供一个量化工具,以了解媒体对其关注的方向和强度。 本数据含原始数据、参考文献、代码do文件、最终结果。参考文献中JF系数计算公式。 指标 代码、年份、标题出现该公司的新闻总数、内容出现该公司的新闻总数、正面新闻数全部、中性新闻数全部、负面新闻数全部、正面新闻数原创、中性新闻数原创、负面新闻数原创,媒体监督JF系数。
AB PLC例程代码项目案例 【备注】 1、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用!有问题请及时沟通交流。 2、适用人群:计算机相关专业(如计科、信息安全、数据科学与大数据技术、人工智能、通信、物联网、自动化、电子信息等)在校学生、专业老师或者企业员工下载使用。 3、用途:项目具有较高的学习借鉴价值,不仅适用于小白学习入门进阶。也可作为毕设项目、课程设计、大作业、初期项目立项演示等。 4、如果基础还行,或热爱钻研,亦可在此项目代码基础上进行修改添加,实现其他不同功能。 欢迎下载!欢迎交流学习!不清楚的可以私信问我!
AB PLC例程代码项目案例 【备注】 1、该资源内项目代码都经过测试运行成功,功能ok的情况下才上传的,请放心下载使用!有问题请及时沟通交流。 2、适用人群:计算机相关专业(如计科、信息安全、数据科学与大数据技术、人工智能、通信、物联网、自动化、电子信息等)在校学生、专业老师或者企业员工下载使用。 3、用途:项目具有较高的学习借鉴价值,不仅适用于小白学习入门进阶。也可作为毕设项目、课程设计、大作业、初期项目立项演示等。 4、如果基础还行,或热爱钻研,亦可在此项目代码基础上进行修改添加,实现其他不同功能。 欢迎下载!欢迎交流学习!不清楚的可以私信问我!