Android 进程间通信(AIDL)使用详解

AIDL简介
AIDL是AndroidInterface definition language的缩写,它是一种Android内部进程通信接口的描述语言。

AIDL使用场景
例如有两个应用应用A和应用K,我们在应用A中绑定了一个Service,并且给对应的数据成员设置了一些值。而我们想要在应用K中得到应用A设置的值,Android平台是不允许不同进程间进行直接通讯的而且不同进程之间也不能通过共享内存的方式进行数据通讯,为了使其他的应用K可以访问应用A提供的服务,AIDL就应运而生。

AIDL使用注意事项
AIDL只支持方法,不能定义静态成员,并且方法也不能有类似public等的修饰符;AIDL运行方法有任何类型的参数和返回值,AIDL支持的数据类型有Java基本数据类型、String、Map、List。List中的所有元素必须是AIDL支持的类型之一,或者是一个其他AIDL生成的接口,或者是定义的Parcelable,List可以使用泛型。Map中的所有元素必须是AIDL支持的类型之一,或者是一个其他AIDL生成的接口,或者是定义的Parcelable,Map是不支持泛型的。

应用A创建AIDL的步骤
1.新建aidl文件夹,如下图:

将焦点移至app上,右键单击,然后New->Folder->AIDL

2.新建*.aidl文件,如下图:
将焦点移至aidl文件夹上,右键单击,然后New->AIDL->AIDL File

3.定义生成的aidl文件内容,代码定义如下:

interface IMyAidlInterface {  
    void basicTypes(int anInt, long aLong, boolean aBoolean, float aFloat, double aDouble, String aString);
    int getanInt();
    long getaLong();
    boolean getaBoolean();
    float getaFloat();
    double getaDouble();
    String getaString();
}

4.继承AIDL文件生成的Java接口Stub,并添加相应的代码实现:

public class BaseDataType extends IMyAidlInterface.Stub {
    int anInt;
    long aLong;
    boolean aBoolean;
    float aFloat;
    double aDouble;
    String aString;
    @Override
    public void basicTypes(int anInt, long aLong, boolean aBoolean, float aFloat, double aDouble, String aString) throws RemoteException {
        this.anInt = anInt;
        this.aLong = aLong;
        this.aBoolean = aBoolean;
        this.aFloat = aFloat;
        this.aDouble = aDouble;
        this.aString = aString;
    }
    @Override
    public int getanInt() throws RemoteException {
        return anInt;
    }
    @Override
    public long getaLong() throws RemoteException {
        return aLong;
    }
    @Override
    public boolean getaBoolean() throws RemoteException {
        return aBoolean;
    }
    @Override
    public float getaFloat() throws RemoteException {
        return aFloat;
    }
    @Override
    public double getaDouble() throws RemoteException {
        return aDouble;
    }
    @Override
    public String getaString() throws RemoteException {
        return aString;
    }
}

PS:IMyAidlInterface.aidl与BaseDataType.java的包名应当是一样的。若想要不一样,请按以下修改build.gradle文件,在 android{} 中间加上下面的内容

sourceSets {
 main { java.srcDirs = ['src/main/java', 'src/main/aidl']
 }
}

5.定义一个Service,并将其注册到AndroidManifest.xml文件中:

import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
public class MyRemoteService extends Service {
    IMyAidlInterface.Stub iStub = new BaseDataType();
    public MyRemoteService() {
    }
    @Override
    public IBinder onBind(Intent intent) {
        return iStub;
    }
}
<service android:name=".MyRemoteService"
    android:enabled="true"
    android:exported="true">
  <intent-filter>
    <action android:name="MyRemoteAIDL" > </action>
  </intent-filter>
</service>

6.在应用A内设置值:

public class MyRemoteActivity extends AppCompatActivity {
    IMyAidlInterface iMyAidlInterface;
    View bind;
    View unbind;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_my_remote);
        Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
        setSupportActionBar(toolbar);
        FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab);
        fab.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                Snackbar.make(view, "Replace with your own action", Snackbar.LENGTH_LONG).setAction("Action", null).show();
            }
        });
        bind = findViewById(R.id.bind);
        unbind = findViewById(R.id.unbind);
        bind.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent intent = new Intent(MyRemoteActivity.this, MyRemoteService.class);
               bindService(intent, conn, Context.BIND_AUTO_CREATE);
            }
        });
    }
    ServiceConnection conn = new ServiceConnection() {
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            iMyAidlInterface = IMyAidlInterface.Stub.asInterface(service);
            if (iMyAidlInterface != null) {
                try {
                    iMyAidlInterface.basicTypes(1024, 2048L, true, 3.14159F, 3.1415926D, "I come from MyRemoteService");
                    android.util.Log.d("RemoteValue", "Set value success!");
                } catch (Exception e) {
                   android.util.Log.d("RemoteValue", "Set value fail!");
                }
            } 
       } 
       @Override
        public void onServiceDisconnected(ComponentName name) {
            android.util.Log.d("RemoteValue", "onServiceDisconnected " + name);
        }
    };
}

至此,同一应用内调用已经完成。
接下来就是被其它应用调用,也就是我们实例中的应用K调用。
应用K访问应用A中的数据步骤
1.将应用A的aidl文件拷贝一份到应用K的程序中(这里一定要注意,包路径要和应用A中的保持一致):

和应用A中的一模一样,直接将其复制到应用K下边即可

2.应用K访问应用A中的数据

public class MainActivity extends Activity {
    IMyAidlInterface myAidlInterface;
    View bind;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        bind = findViewById(R.id.bind);
        bind.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent intent = new Intent();
                intent.setAction("MyRemoteAIDL");
                //此处设置应用A的包名
                intent.setPackage("qihoo.testcodedemo");
                Intent newIntent = buildExplicitIntent(MainActivity.this, intent);
                boolean isbind = getApplicationContext().bindService(newIntent, connection, Context.BIND_AUTO_CREATE);
            }
        });
    }
    public Intent buildExplicitIntent(Context context, Intent intent) {
        PackageManager pm = context.getPackageManager();
        List<ResolveInfo> resolveInfo = pm.queryIntentServices(intent, 0);
        if (resolveInfo == null || resolveInfo.size() != 1) {
            return null;
        }
        // Get component info and create ComponentName
        ResolveInfo serviceInfo = resolveInfo.get(0);
        String packageName = serviceInfo.serviceInfo.packageName;
        String className = serviceInfo.serviceInfo.name;
        ComponentName component = new ComponentName(packageName, className);
        // Create a new intent. Use the old one for extras and such reuse
        Intent explicitIntent = new Intent(intent);
        // Set the component to be explicit
        explicitIntent.setComponent(component);
        return explicitIntent;
    }
    ServiceConnection connection = new ServiceConnection() {
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            myAidlInterface = IMyAidlInterface.Stub.asInterface(service);
            if (myAidlInterface != null) {
                try {
                    android.util.Log.e("RemoteService", " " + myAidlInterface.getanInt());
                    android.util.Log.e("RemoteService", myAidlInterface.getaString());
                } catch (Exception e) {
                }
            }
        }
        @Override
        public void onServiceDisconnected(ComponentName name) {
        }
    };
}

PS:将两个App同时运行在同一台手机上,可访问正确的数据值;
至此,通过AIDL的方式进行两个不同App的数据访问已完成;
敬请各位童鞋关注本人微信公众号“IT软件人”,或扫码关注。

IT软件人

热烈欢迎与各位童鞋进行技术交流。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,012评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,628评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,653评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,485评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,574评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,590评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,596评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,340评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,794评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,102评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,276评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,940评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,583评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,201评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,441评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,173评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,136评论 2 352

推荐阅读更多精彩内容