AIDL Service,跨进程调用Services

一、AIDL Service简介
Android系统中,各个应用都运行在自己的进程中,进程之间一般无法直接进行通信,为了实现进程通信,Android提供了AIDL Service;

二、与本地Service不同 

本地Service:直接把IBinder对象本身传递给客户端的ServiceConnection的onServiceConnected方法的第二个参数; 
远程Service:只将IBinder对象的代理传给客户端的ServiceConnection的onServiceConnected方法的第二个参数; 
三、AIDL文件
Android需要AIDL(Android Interface Definition Language)来定义远程接口,这种接口定义语言并不是一种真正的变成语言,只是定义两个进程之间的通信接口;
与Java接口相似,但是存在如下几点差异:
AIDL定义接口的源代码必须以.aidl结尾;  
AIDL用到的数据类型,除了基本类型、String、List、Map、CharSequence之外,其它类型全部都需要导包,即使它们在同一个包中也需要导包;
四、例子:

1. 创建AIDL文件,定义好的AIDL文件后,ADT工具会自动在gen目录下生成一个AIDL.java接口,该类内部包含一个Stub内部类,实现了IBinder,AIDL里面的接口,这个Stub类会作为远程Service回调类:

IMyService.aidl
package com.juno.serviceaidltest;  import com.juno.serviceaidltest.Product;  
interface IMyService    
{    String getValue();  Map getMap(in String country, in Product product);    Product getProduct();  
}  

Product.aidl
parcelable Product; 

Product.java
package com.juno.serviceaidltest;  import android.os.Parcel;  
import android.os.Parcelable;  public class Product implements Parcelable {  private int id;  private String name;  private float price;  public static final Parcelable.Creator<Product> CREATOR = new Parcelable.Creator<Product>() {  public Product createFromParcel(Parcel in) {  return new Product(in);  }  public Product[] newArray(int size) {  return new Product[size];  }  };  public Product() {  }  private Product(Parcel in) {  readFromParcel(in);  }  @Override  public int describeContents() {  return 0;  }  public void readFromParcel(Parcel in) {  id = in.readInt();  name = in.readString();  price = in.readFloat();  }  @Override  public void writeToParcel(Parcel dest, int flags) {  dest.writeInt(id);  dest.writeString(name);  dest.writeFloat(price);  }  public int getId() {  return id;  }  public void setId(int id) {  this.id = id;  }  public String getName() {  return name;  }  public void setName(String name) {  this.name = name;  }  public float getPrice() {  return price;  }  public void setPrice(float price) {  this.price = price;  }  }  

2. 将接口暴露给客户端

MyService.java
package com.juno.serviceaidltest;  import java.util.HashMap;  
import java.util.Map;  import android.app.Service;  
import android.content.Intent;  
import android.os.IBinder;  
import android.os.RemoteException;  public class MyService extends Service {  /** * 继承Stub,也就是实现了IMyService接口,并实现了IBinder接口 */  public class MyServiceImpl extends IMyService.Stub {  @Override  public String getValue() throws RemoteException {  return "Test Value";  }  @Override  public Map<String, Object> getMap(String country, Product product)  throws RemoteException {  Map<String, Object> map = new HashMap<String, Object>();  map.put("country", country);  map.put("id", product.getId());  map.put("name", product.getName());  map.put("price", product.getPrice());  map.put("product", product);  return map;  }  @Override  public Product getProduct() throws RemoteException {  Product product = new Product();  product.setId(1234);  product.setName("汽车");  product.setPrice(31000);  return product;  }  }  @Override  public IBinder onBind(Intent intent) {  /** * 返回MyServiceImpl对象,在绑定本地Service情况下,该MyServiceImpl会直接传给客户端的ServiceConnected对象的ServiceConnected()方法的第二个参数;在绑定远程Service的情况下,只将MyServiceImpl对象的代理传给客户端的ServiceConnected对象的ServiceConnected()方法的第二个参数 */  return new MyServiceImpl();  }  }  

3.  在AndroidManifext.xml文件中配置该Service:

<service android:name=".MyService" >  <intent-filter>  <action android:name="com.juno.serviceaidltest.IService" />  </intent-filter>  </service>

4. 在Activity里访问 AIDLService,如果不在同一个App下面访问,需要将Service端的AIDL文件复制到客户端中,并在相同的包名下

MainActivity.java
package com.juno.serviceanotheraidltest;  import android.app.Activity;  
import android.content.ComponentName;  
import android.content.Context;  
import android.content.Intent;  
import android.content.ServiceConnection;  
import android.os.Bundle;  
import android.os.IBinder;  
import android.os.RemoteException;  
import android.util.Log;  
import android.view.View;  
import android.widget.Button;  
import android.widget.TextView;  import com.juno.serviceaidltest.IMyService;  public class MainActivity extends Activity implements View.OnClickListener {  private final static String ACTION = "com.juno.serviceaidltest.IService";  private IMyService myService = null;  private Button mBtnInvokeAIDLService;  private Button mBtnBindAIDLService;  private TextView mTextView;  private ServiceConnection mServiceConnection = new ServiceConnection() {  @Override  public void onServiceConnected(ComponentName name, IBinder service) {  //获取远程Service的onBinder方法返回的对象代理  myService = IMyService.Stub.asInterface(service);  mBtnInvokeAIDLService.setEnabled(true);  try {  Log.v("juno", myService.getValue());  } catch (RemoteException e) {  e.printStackTrace();  }  }  @Override  public void onServiceDisconnected(ComponentName name) {  myService = null;  }  };  @Override  protected void onCreate(Bundle savedInstanceState) {  super.onCreate(savedInstanceState);  setContentView(R.layout.activity_main);  initView();  }  private void initView() {  mBtnInvokeAIDLService = (Button) findViewById(R.id.btnInvokeAIDLService);    mBtnBindAIDLService = (Button) findViewById(R.id.btnBindAIDLService);    mBtnInvokeAIDLService.setEnabled(false);    mTextView = (TextView) findViewById(R.id.textView1);    mBtnInvokeAIDLService.setOnClickListener(this);    mBtnBindAIDLService.setOnClickListener(this);         }  @Override  public void onClick(View view) {  switch (view.getId()) {  case R.id.btnBindAIDLService:  //创建所需要绑定的Service的Intent,绑定远程的服务  bindService(new Intent(ACTION), mServiceConnection, Context.BIND_AUTO_CREATE);  break;  case R.id.btnInvokeAIDLService:  try {  String s = myService.getValue();    s = "Product.id = " + myService.getProduct().getId() + "\n";    s += "Product.name = " + myService.getProduct().getName()    + "\n";    s += "Product.price = " + myService.getProduct().getPrice()    + "\n";    s += myService.getMap("China", myService.getProduct()).toString();   mTextView.setText(myService.asBinder().isBinderAlive() + "  " + s);  } catch (Exception e) {  }  break;  }  }  @Override  protected void onDestroy() {  super.onDestroy();  if (myService != null) {  //解除绑定  unbindService(mServiceConnection);  }  }  }  

布局文件

activity_main.xml
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"  xmlns:tools="http://schemas.android.com/tools"  android:layout_width="match_parent"  android:layout_height="match_parent"  android:paddingBottom="@dimen/activity_vertical_margin"  android:paddingLeft="@dimen/activity_horizontal_margin"  android:paddingRight="@dimen/activity_horizontal_margin"  android:paddingTop="@dimen/activity_vertical_margin"  tools:context=".MainActivity" >  <TextView  android:id="@+id/textView1"  android:layout_width="wrap_content"  android:layout_height="wrap_content"  android:text="@string/hello_world" />  <Button  android:id="@+id/btnBindAIDLService"  android:layout_width="wrap_content"  android:layout_height="wrap_content"  android:layout_below="@id/textView1"  android:text="btnBindAIDLService" />  <Button  android:id="@+id/btnInvokeAIDLService"  android:layout_width="wrap_content"  android:layout_height="wrap_content"  android:layout_below="@id/btnBindAIDLService"  android:text="btnInvokeAIDLService" />  </RelativeLayout>  

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/430327.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

一台电脑同时启动多个java_如何在一台同时登录多个微信?

点击上方蓝字关注我相信大家对手机上的微信双开并不陌生&#xff0c;很多人由于工作需要&#xff0c;需要多开一个甚至多个微信。对于安卓手机来说&#xff0c;双开微信并不是什么难事。现在大部分国产安卓手机都自带微信双开功能(一般在设置里面搜索“分身”或者“双开”就能找…

利用Python进行数据分析(1) 简单介绍

一、处理数据的基本内容 数据分析 是指对数据进行控制、处理、整理、分析的过程。 在这里&#xff0c;“数据”是指结构化的数据&#xff0c;例如&#xff1a;记录、多维数组、Excel 里的数据、关系型数据库中的数据、数据表等。 二、说说 Python 这门语言 Python 是现在最受…

python每周小测验答案_python第一周小测验答案Centos下更新Python版本

Centos下更新Python版本&#xff0c;有需要的朋友可以参考下。 注意&#xff1a;更新python千万不要把老版本的删除&#xff01;新老版本是可以共存的&#xff0c;很多基本的命令、软件包都要依赖预装的老版本python的&#xff0c;比如yum。 更新python&#xff1a; 第1步&…

【复习】使用 SQLiteDatabase 操作 SQLite 数据库

Android提供了一个名为SQLiteDatabase的类&#xff0c;该类封装了一些操作数据库的API&#xff0c;使用该类可以完成对数据进行添加(Create)、查询(Retrieve)、更新(Update)和删除(Delete)操作&#xff08;这些操作简称为CRUD&#xff09;。对SQLiteDatabase的学习&#xff0c;…

grep, egrep, fgrep笔记

grep, egrep, fgrep grep: 根据模式搜索文本&#xff0c;并将符合模式的文本行显示出来。Pattern: 文本字符和正则表达式的元字符组合而成匹配条件grep [options] PATTERN [FILE...] -i&#xff1a;不区分大小写 --color -v: 显示没有被模式匹配到的行 -o&#…

python可以开多少线程_python多线程详解

python多线程详解 一、线程介绍 什么是线程 线程&#xff08;Thread&#xff09;也叫轻量级进程&#xff0c;是操作系统能够进行运算调度的最小单位&#xff0c;它被包涵在进程之中&#xff0c;是进程中的实际运作单位。线程自己不拥有系统资源&#xff0c;只拥有一点儿在运行中…

android 设置线程优先级

1. android.os.Process.setThreadPriority &#xff08;int priority&#xff09;或 android.os.Process.setThreadPriority &#xff08;int tid&#xff0c; int priority&#xff09; priority&#xff1a;【-20&#xff0c; 19】&#xff0c;高优先级 -> 低优先级…

ArcGIS for Desktop入门教程_第八章_Desktop学习资源 - ArcGIS知乎-新一代ArcGIS问答社区...

ArcGIS for Desktop入门教程_第八章_Desktop学习资源 - ArcGIS知乎-新一代ArcGIS问答社区 原文:ArcGIS for Desktop入门教程_第八章_Desktop学习资源 - ArcGIS知乎-新一代ArcGIS问答社区1 学习资源用户在学习和应用过程中&#xff0c;可以参考的资源如下&#xff1a;1. ArcGIS资…

python中面向对象_简述Python中的面向对象编程的概念

面向对象编程——Object Oriented Programming&#xff0c;简称OOP&#xff0c;是一种程序设计思想。OOP把对象作为程序的基本单元&#xff0c;一个对象包含了数据和操作数据的函数。 面向过程的程序设计把计算机程序视为一系列的命令集合&#xff0c;即一组函数的顺序执行。为…

android 使用Binder通信

遇到下雨&#xff0c;写个简单例子&#xff0c;用于Activity与Service通信&#xff0c;之前也有一篇Binder的&#xff1a;点击。 例子非常简单&#xff1a; MainActivity.java package com.juno.testbind;import android.app.Activity; import android.content.ComponentNam…

随机数测试

学习链接 http://lavasoft.blog.51cto.com/62575/113758/import java.util.Random; /** * 随机数、随即字符串工具 * User: leizhimin * Date: 2008-11-19 9:43:09 */ public class RandomUtils { public static final String allChar "0123456789abcdefghijklmnopq…

python spark dataframe_pyspark dataframe 常用操作

spark dataframe派生于RDD类&#xff0c;但是提供了非常强大的数据操作功能。当然主要对类SQL的支持。 在实际工作中会遇到这样的情况&#xff0c;主要是会进行两个数据集的筛选、合并&#xff0c;重新入库。 首先加载数据集&#xff0c;然后在提取数据集的前几行过程中&#x…

Android使用PorterDuffXfermode实现遮罩效果

当我们在程序里需要做遮罩效果时&#xff0c;可以PorterDuffXfermode类实现。 例如有一副图片&#xff0c;需要使用一个图形&#xff08;一般是可以使用Canvas画出来的图形&#xff09;&#xff0c;图形覆盖到的地方在图片上才显示出来&#xff0c;可以使用PorterDuff.Mode.SR…

Oracle数据库sql调优

最近做的一个项目涉及到大数据量的数据表查询。数据表总量在7亿-20亿左右&#xff0c;以主键建立全局唯一索引&#xff0c;分区策略是hash分区范围分区&#xff0c;很大部分情况下查询条件命中记录数超过万条&#xff0c;单次返回前XX条记录。在调优过程中总结出以下经验&#…

amd sata controller下载_AMD发布全新锐龙芯片组驱动:告别卡死、报错

AMD今天发布了全新的锐龙平台芯片组驱动&#xff0c;版本号2.04.04.111&#xff0c;从界面到功能都全面升级&#xff0c;并修复了此前存在的多个严重Bug。新驱动重新设计了安装界面&#xff0c;借鉴了Radeon Adrenalin 2020肾上腺素显卡驱动的诸多元素&#xff0c;更加时尚美观…

Android自定义属性动画

当我们对一个View进行透明度变化的动画&#xff0c;我们可以这样写&#xff1a; ValueAnimator fadeAnimator ObjectAnimator.ofFloat(myView, View.ALPHA, 1.0f, 0.0f); fadeAnimator.setInterpolator(PathInterpolatorCompat.create(0.33f, 0f, 0.66f, 1f)); fadeAnimator.s…

mysql数据库基础的简单操作指南

最近在学习mysql,本文是做的关于mysql学习的笔记&#xff0c;跟大家分享一下&#xff0c;希望对大家学习mysql知识有所助益。mysql现在几乎已经成了网站建设的主流数据库&#xff0c;很多php网站系统都采用了mysql数据库。比mssql最大的优势在于开源&#xff0c;免费。希望与大…

kill 进程_结束进程,查看后台进程

有一些进程想要结束怎么办呢&#xff1f;进程的优先是怎样的呢?如何查看后台正在运行的进程呢&#xff1f;下面我就来讲讲给进程发送信号(kill -l列出所有支持的信号)[rootqianfeng ~]# kill -l 编号 信号名1) SIGHUP 重新加载配置2) SIGINT 键盘中断CtrlC3) SIGQUIT 键盘退出…

软碟通

转载于:https://www.cnblogs.com/jcfxl/p/5739741.html

AlarmManager使用

AlarmManager是Android中常用的一种系统级别的提示服务&#xff0c;在特定的时刻为我们广播一个指定的Intent&#xff0c;为你的应用设定一个在未来某个时间唤醒的功能。 当闹钟响起&#xff0c;实际上是系统发出了为这个闹钟注册的广播&#xff0c;会自动开启目标应用。 注册…