如有转载,请申明:
转载至 http://blog.csdn.net/qq_35064774/article/details/52594854
在安卓中不仅支持应用内Service绑定,还支持远程Service绑定。
在eclipse中很方便就可以实现,但在Android Studio中有些不同,下面完整介绍在Android Studio中远程Service的编写和调用。
在Eclipse中很简单,只需要先编写一个接口文件,然后把.java改成.aidl即可自动识别并解释出相应的.java文件。
然而在Android Studio中则并不是这样随意。我们选中需要新建AIDL的项目,右键弹出菜单,如图,选择New -> AIDL -> AIDL File。
建好文件后,我们在里面编写需要的接口方法,注意,不需要加限定修饰符(public private这些)。
接口写好并不代表就完成了,你还得让IDE能够找到你的自定义接口。
在我这个版本的Android Studio中,并不能自动找到自定义接口,所以还需要进行如下配置(如果你的可以在Service类中直接使用,那就没必要进行这一步)
打开项目对应的build.gradle,注意是后面括号中有Modulle的。
在这个文件中新增如下代码:
sourceSets {
main {
manifest.srcFile 'src/main/AndroidManifest.xml'
java.srcDirs = ['src/main/java', 'src/main/aidl']
resources.srcDirs = ['src/main/java', 'src/main/aidl']
aidl.srcDirs = ['src/main/aidl']
res.srcDirs = ['src/main/res']
assets.srcDirs = ['src/main/assets']
}
}
为了防止新手写错位置,把完整的配置内容也贴出来。在修改后,代码如下:
apply plugin: 'com.android.application'
android {
compileSdkVersion 23
buildToolsVersion "23.0.3"
defaultConfig {
applicationId "com.ittianyu.basictest"
minSdkVersion 15
targetSdkVersion 23
versionCode 1
versionName "1.0"
}
sourceSets {
main {
manifest.srcFile 'src/main/AndroidManifest.xml'
java.srcDirs = ['src/main/java', 'src/main/aidl']
resources.srcDirs = ['src/main/java', 'src/main/aidl']
aidl.srcDirs = ['src/main/aidl']
res.srcDirs = ['src/main/res']
assets.srcDirs = ['src/main/assets']
}
}
buildTypes {
release {
minifyEnabled false
proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
}
}
}
dependencies {
compile fileTree(dir: 'libs', include: ['*.jar'])
testCompile 'junit:junit:4.12'
compile 'com.android.support:appcompat-v7:23.4.0'
compile 'com.android.support:design:23.4.0'
}
目的是让gradle把aidl文件也自动包含进来。然后通过菜单命令清理工程或者重新建立工程。
这一步完成后,AIDL的编写就正式完成。
支持远程调用的Service与普通的Service基本一样,唯一的区别是,onBind返回的代理类,这个类需要继承自己编写的AIDL接口中的Stub类,比如我这里是IService.Stub。为了方便看,我把完整代码贴出来。(别忘了配置AndroidManifest.xml的action)
package com.ittianyu.remoteservice;
import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
public class RemoteService extends Service {
@Override
public IBinder onBind(Intent intent) {
return new Agent();
}
/**
* 自定义服务方法
*/
public void method() {
System.out.println("服务中的方法被调用了");
}
@Override
public void onCreate() {
super.onCreate();
System.out.println("远程服务创建了");
}
@Override
public void onDestroy() {
super.onDestroy();
System.out.println("远程服务销毁了");
}
/**
* 内部代理人
*/
private class Agent extends IService.Stub{
@Override
public void callMethodInService() {
method();
}
}
}
如果你在这一步发现你自己编写的AIDL类无法识别,比如我这里的是IService.Stub,请对照AIDL的编写中的最后一步,看看是不是配置错了。
到这里,远程服务端的代码编写好了。
为了保持接口类一致,我们需要切换到Project或者其他可以看到src下目录结果的视图
把服务端的aidl文件夹复制到调用端工程的src/main目录下。(什么,你问我服务端的aidl文件夹在哪?切换到Project视图,src/main下就有)
直接复制aidl文件夹的目的是保持接口的完整包名一致。
除了可以像上面这样在IDE中直接操作,你也可以直接找到工程目录,手动复制粘贴到对应位置,然后Clean Project就自动识别了。
最后还有一步,也是关于AIDL接口的识别问题,同意需要在gradle中配置,具体步骤和前者一样,这里不重复了。
调用类和普通的本地调用基本差不多,唯一的区别是ServiceConnection实现类中的方法onServiceConnected的具体实现,如果是本地调用,我们可以直接把参数service强制转成接口类型,但在远程调用中,需要调用AIDL中的方法来转换。
比如:agent = IService.Stub.asInterface(service);
为了便于查看,我也把完整代码贴出来。
package com.ittianyu.basictest.callremoteservice;
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.support.v7.app.AppCompatActivity;
import android.view.View;
import android.widget.Toast;
import com.ittianyu.remoteservice.IService;
import com.ittianyu.basictest.R;
public class CallRemoteServiceActivity extends AppCompatActivity implements View.OnClickListener {
private MyConnection connection;
private IService agent;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_call_remote_service);
init();
}
/**
* 控件事件绑定
*/
private void init() {
findViewById(R.id.btn_bind_reomte_service).setOnClickListener(this);
findViewById(R.id.btn_unbind_remote_service).setOnClickListener(this);
findViewById(R.id.btn_call_remote_service_method).setOnClickListener(this);
}
/**
* 点击事件
* @param v
*/
@Override
public void onClick(View v) {
switch (v.getId()) {
case R.id.btn_bind_reomte_service:
bindRemoteService();
break;
case R.id.btn_unbind_remote_service:
unbindRemoteService();
break;
case R.id.btn_call_remote_service_method:
callRemoteServiceMethod();
break;
}
}
/**
* 调用远程服务方法
*/
private void callRemoteServiceMethod() {
if(null == agent) {
Toast.makeText(CallRemoteServiceActivity.this, "还未绑定服务", Toast.LENGTH_SHORT).show();
return;
}
try {
agent.callMethodInService();
} catch (RemoteException e) {
e.printStackTrace();
Toast.makeText(CallRemoteServiceActivity.this, "调用失败", Toast.LENGTH_SHORT).show();
return;
}
}
/**
* 解绑远程服务
*/
private void unbindRemoteService() {
if(null == connection) {
Toast.makeText(CallRemoteServiceActivity.this, "服务还未绑定", Toast.LENGTH_SHORT).show();
return;
}
unbindService(connection);
connection = null;
agent = null;
Toast.makeText(CallRemoteServiceActivity.this, "解绑成功", Toast.LENGTH_SHORT).show();
}
/**
* 绑定远程服务
*/
private void bindRemoteService() {
if(null != connection) {
Toast.makeText(CallRemoteServiceActivity.this, "已经绑定了服务", Toast.LENGTH_SHORT).show();
return;
}
connection = new MyConnection();
Intent intent = new Intent("com.ittianyu.service.remote.test");
if(!bindService(intent, connection, Context.BIND_AUTO_CREATE)) {
Toast.makeText(CallRemoteServiceActivity.this, "绑定失败", Toast.LENGTH_SHORT).show();
return;
}
Toast.makeText(CallRemoteServiceActivity.this, "绑定成功", Toast.LENGTH_SHORT).show();
}
private class MyConnection implements ServiceConnection {
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
agent = IService.Stub.asInterface(service);
}
@Override
public void onServiceDisconnected(ComponentName name) {
agent = null;
}
}
}
到此,填坑完毕= =。