Android使用AIDL跨進程通信


一、基本類型

1、AIDL是什么

AIDL是Android中 IPC(Inter-Process Communication)方式中的一種,AIDL是 Android Interface definition language的縮寫,對於小白來說,AIDL的作用是讓你可以在自己的APP里綁定一個其他APP的service,這樣你的APP可以和其他APP交互。

2、AIDL的使用

因為是兩個app交互,所以需要創建兩個app。

 

1.創建AIDL文件

在app項目創建aidl文件

輸入名稱后,as就幫我們創建了一個AIDL文件。

 

上面就是as幫我生成的aidl文件。basicTypes這個方法可以無視,看注解知道這個方法只是告訴你在AIDL中你可以使用的基本類型(int, long, boolean, float, double, String),因為這里是要跨進程通訊的,所以不是隨便你自己定義的一個類型就可以在AIDL使用的,這些后面會說。我們在AIDL文件中定義一個我們要提供給第二個APP使用的接口。

定義好之后,就可以sycn project一下,然后新建一個service。在service里面創建一個內部類,繼承你剛才創建的AIDL的名稱里的Stub類,並實現接口方法,在onBind返回內部類的實例。

import android.os.RemoteException;

public class MyService extends Service {
    public MyService() {
    }

    @Override
    public IBinder onBind(Intent intent) {
        // TODO: Return the communication channel to the service.
        return new MyBinder();
    }

    class MyBinder extends Person.Stub {

        @Override
        public String getName() throws RemoteException {
            return "456";
        }
    }
}

接下來,將我們的AIDL文件拷貝到第二個項目(aidlcalldemo),然后sycn project一下工程。

 

注意:這邊的包名要跟第一個項目的一樣哦,這之后在Activity中綁定服務。

package com.lyf.test.aidlcalldemo;

import android.content.ComponentName;
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.widget.Button;
import android.widget.Toast;

import com.lyf.test.aidldemo.Person;

import butterknife.BindView;
import butterknife.ButterKnife;
import butterknife.OnClick;

public class MainActivity extends AppCompatActivity {

    @BindView(R.id.button)
    Button button;
    private Person person;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        ButterKnife.bind(this);
        Intent intent = new Intent();
        intent.setComponent(new ComponentName("com.lyf.test.aidldemo", "com.lyf.test.aidldemo.MyService"));
        bindService(intent, new ServiceConnection() {
            @Override
            public void onServiceConnected(ComponentName name, IBinder service) {
                person = Person.Stub.asInterface(service);
            }

            @Override
            public void onServiceDisconnected(ComponentName name) {

            }
        }, BIND_AUTO_CREATE);
    }

    @OnClick(R.id.button)
    public void onViewClicked() {
        try {
            Toast.makeText(MainActivity.this, person.getName(), Toast.LENGTH_SHORT).show();
        } catch (RemoteException e) {
            e.printStackTrace();
        }
    }
}

Android5.0之后只能使用顯式Intent啟動Service組件。

在onServiceConnected方法中通過Person.Stub.asInterface(service)獲取Person對象,然后在onClick中調用person.getName()。

 

 二、自定義類型

如果我要在AIDL中使用自定義的類型,要怎么做呢。首先我們的自定義類型要實現 Parcelable接口,下面的代碼中創建了一個User類並實現Parcelable接口。這邊就不對Parcelable進行介紹了,不熟悉的童鞋自行查找資料,總之我們這邊可以借助studio的Show Intention Action(也就是Eclipse中的Quick Fix,默認是alt+enter鍵)幫我們快速實現Parcelable接口。
 
1、首先在一個包中創建java類

實現Parcelable接口

接下新建一個aidl文件,名稱為我們自定義類型的名稱,這邊是User.aidl。

 

在User.aidl聲明我們的自定義類型和它的完整包名,注意這邊parcelable是小寫的,不是Parcelable接口,一個自定類型需要一個這樣同名的AIDL文件。

package com.lyf.test.aidldemo;
parcelable User;
然后再在我們的AIDL接口中導入我們的AIDL類型。

然后定義接口方法,sycn project后就可以在service中做具體實現了。

 
package com.lyf.test.aidldemo;

import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.os.RemoteException;

public class MyService extends Service {
    public MyService() {
    }

    @Override
    public IBinder onBind(Intent intent) {
        // TODO: Return the communication channel to the service.
        return new MyBinder();
    }

    class MyBinder extends IMyAidlInterface.Stub {

        @Override
        public String getName() throws RemoteException {
            return "456";
        }

        @Override
        public User getUserName() throws RemoteException {
            return new User("789");
        }
    }
}

最后將我們的AIDL文件和自定義類型的java一並拷貝到第二個項目,注意包名都要一樣哦。

然后就可以在Activity中使用該自定義類型的AIDL接口了。

 

package com.lyf.test.aidlcalldemo;

import android.content.ComponentName;
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.widget.Button;
import android.widget.Toast;

import com.lyf.test.aidldemo.IMyAidlInterface;

import butterknife.BindView;
import butterknife.ButterKnife;
import butterknife.OnClick;

public class MainActivity extends AppCompatActivity {

    @BindView(R.id.button)
    Button button;
    private IMyAidlInterface iMyAidlInterface;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        ButterKnife.bind(this);
        Intent intent = new Intent();
        intent.setComponent(new ComponentName("com.lyf.test.aidldemo", "com.lyf.test.aidldemo.MyService"));
        bindService(intent, new ServiceConnection() {
            @Override
            public void onServiceConnected(ComponentName name, IBinder service) {
                iMyAidlInterface = IMyAidlInterface.Stub.asInterface(service);
            }

            @Override
            public void onServiceDisconnected(ComponentName name) {

            }
        }, BIND_AUTO_CREATE);
    }

    @OnClick(R.id.button)
    public void onViewClicked() {
        try {
            Toast.makeText(MainActivity.this, iMyAidlInterface.getUserName().getName(), Toast.LENGTH_SHORT).show();
        } catch (RemoteException e) {
            e.printStackTrace();
        }
    }
}

大功告成。

github項目demo:https://github.com/First-Time/AIDLDemo.git


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM