深入学习Android——AIDL机制

来源:互联网 发布:登录我的淘宝 编辑:程序博客网 时间:2024/05/22 13:17

简洁起见,把这2篇文章分开写,由浅入深,学习本来就是一个积累的过程嘛!

上一篇通过AIDL传递了一个基本的数据类型,String跑完整个demo通过log就可以看出来了,我传递的数据是字符串,为什么不传递其他的数据试一试呢,到底能传递什么类型的数据呢,你清楚吗?



AIDL 使用简单语法,使您能通过可带参数和返回值的一个或多个方法来声明接口。 参数和返回值可以是任意类型,甚至可以是其他 AIDL 生成的接口。您必须使用 Java 编程语言构建 .aidl 文件。每个 .aidl 文件都必须定义单个接口,并且只需包含接口声明和方法签名。


默认情况下,AIDL 支持下列数据类型:
Java 编程语言中的所有原语类型(如 int、long、char、boolean 等等)
String
CharSequence
List
List 中的所有元素都必须是以上列表中支持的数据类型、其他 AIDL 生成的接口或您声明的可打包类型。 可选择将 List 用作“通用”类(例如,List<String>)。另一端实际接收的具体类始终是 ArrayList,但生成的方法使用的是 List 接口。
Map
Map 中的所有元素都必须是以上列表中支持的数据类型、其他 AIDL 生成的接口或您声明的可打包类型。 不支持通用 Map(如 Map<String,Integer> 形式的 Map)。 另一端实际接收的具体类始终是 HashMap,但生成的方法使用的是 Map 接口。


好了!支持大部分的数据类型,那么实现跨进程通信就方便了。传递String和上述数据类型一致,这里就不再赘述了,那么如何传递对象呢?你有没有试一试呢?


通过 IPC 传递对象


通过 IPC 接口把某个类从一个进程发送到另一个进程是可以实现的。 不过,您必须确保该类的代码对 IPC 通道的另一端可用,并且该类必须支持Parcelable 接口。支持 Parcelable 接口很重要,因为 Android 系统可通过它将对象分解成可编组到各进程的原语。

意思就是说:activity 和service2端都支持该对象的传递,而这个对象本身应该实现Parcelable这个接口。
参考demo
http://blog.csdn.net/luoyanglizi/article/details/51980630

Book.aidl 该文件简单的2句话就搞定了,package指定所在的包,parcelable指定了Book实现该接口。

// Book.aidlpackage com.example.mytry;parcelable Book;

BookManager.aidl 这个是Book的管理类,包含声明的方法

// BookManager.aidlpackage com.example.mytry;import com.example.mytry.Book;interface BookManager {    List<Book> getBooks();    void addBook(in Book book);}
就这样2个方法getBook()和addBook(),当然Book这个类是我们自己定义的,并且实现了Parcelable接口,它是一个类,不是基本的数据类型哦,在AIDL当中就需要导入import com.example.mytry.Book;

Book.java

package com.example.mytry;import android.os.Parcel;import android.os.Parcelable;public class Book implements Parcelable {    private String name;    private int price;    public Book() {    }    public String getName() {        return name;    }    public void setName(String name) {        this.name = name;    }    public int getPrice() {        return price;    }    public void setPrice(int price) {        this.price = price;    }    public Book(Parcel in) {        name = in.readString();        price = in.readInt();    }    public static final Creator<Book> CREATOR = new Creator<Book>() {        @Override        public Book createFromParcel(Parcel in) {            return new Book(in);        }        @Override        public Book[] newArray(int size) {            return new Book[size];        }    };    @Override    public int describeContents() {        return 0;    }    @Override    public void writeToParcel(Parcel dest, int flags) {        dest.writeString(name);        dest.writeInt(price);    }    //方便打印数据    @Override    public String toString() {        return "name : " + name + " , price : " + price;    }}

AndroidManifest.xml 中记得配置android:process=":remote",不指明这个属性,你的activity就不能绑定到服务

<service    android:name=".MyService2"   android:process=":remote"/>

当然这些搞定了,就涉及activity+service如何实现了,布局加一个按钮即可,指定android:onClick=“addBook”;

MainActivity.java

package com.example.mytry;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.util.Log;import android.view.View;import android.widget.Toast;import java.util.List;public class MainActivity extends AppCompatActivity {    private BookManager mBookManager = null;    private boolean mBound = false;    private List<Book> mBooks;    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_main);    }    /**     * 按钮的点击事件,点击之后调用服务端的addBookIn方法     *     * @param view     */    public void addBook(View view) {        if (!mBound) {            attemptToBindService();            Toast.makeText(this, "当前与服务端处于未连接状态,正在尝试重连,请稍后再试", Toast.LENGTH_SHORT).show();            return;        }        if (mBookManager == null) return;        Book book = new Book();        book.setName("APP研发录In");        book.setPrice(30);        try {            mBookManager.addBook(book);            Log.e(getLocalClassName(), book.toString());        } catch (RemoteException e) {            e.printStackTrace();        }    }    /**     * 尝试与服务端建立连接     */    private void attemptToBindService() {        Intent intent = new Intent(MainActivity.this, MyService2.class);        bindService(intent, mServiceConnection, Context.BIND_AUTO_CREATE);    }    @Override    protected void onStart() {        super.onStart();        if (!mBound) {            attemptToBindService();        }    }    @Override    protected void onStop() {        super.onStop();        if (mBound) {            unbindService(mServiceConnection);            mBound = false;        }    }    private ServiceConnection mServiceConnection = new ServiceConnection() {        @Override        public void onServiceConnected(ComponentName name, IBinder service) {            Log.e(getLocalClassName(), "service connected");            mBookManager = BookManager.Stub.asInterface(service);            mBound = true;            if (mBookManager != null) {                try {                    mBooks = mBookManager.getBooks();                    Log.e(getLocalClassName(), mBooks.toString());                } catch (RemoteException e) {                    e.printStackTrace();                }            }        }        @Override        public void onServiceDisconnected(ComponentName name) {            Log.e(getLocalClassName(), "service disconnected");            mBound = false;        }    };}

MyService2.java 请原谅我随便起了个名字,重要的是内涵。

package com.example.mytry;import android.app.Service;import android.content.Intent;import android.os.IBinder;import android.os.RemoteException;import android.util.Log;import java.util.ArrayList;import java.util.List;public class MyService2 extends Service {    public final String TAG = this.getClass().getSimpleName();    private List<Book> mBooks = new ArrayList<>();        private final BookManager.Stub mBookManager = new BookManager.Stub() {        @Override        public List<Book> getBooks() throws RemoteException {            synchronized (this) {                Log.e(TAG, "invoking getBooks() method , now the list is : " + mBooks.toString());                if (mBooks != null) {                    return mBooks;                }                return new ArrayList<>();            }        }        @Override        public void addBook(Book book) throws RemoteException {            synchronized (this) {                if (mBooks == null) {                    mBooks = new ArrayList<>();                }                if (book == null) {                    Log.e(TAG, "Book is null in In");                    book = new Book();                }                //尝试修改book的参数,主要是为了观察其到客户端的反馈                book.setPrice(2333);                if (!mBooks.contains(book)) {                    mBooks.add(book);                }                //打印mBooks列表,观察客户端传过来的值                Log.e(TAG, "invoking addBooks() method , now the list is : " + mBooks.toString());            }        }    };    @Override    public void onCreate() {        super.onCreate();        Book book = new Book();        book.setName("Android开发艺术探索");        book.setPrice(28);        mBooks.add(book);    }    @Override    public IBinder onBind(Intent intent) {        Log.e(getClass().getSimpleName(), String.format("on bind,intent = %s", intent.toString()));        return mBookManager;    }}

都搞定了,却发现还是绑定不到服务,检查了一遍,遇到了传说中的坑,其实就是几句话的事!

Book.aidl与Book.java的包名应当是一样的,还应该在Module的build.gradle 文件下android{}中指明。

sourceSets {    main {        java.srcDirs = ['src/main/java', 'src/main/aidl']    }}
设置好这一切,就能看到成效了。

再次感谢大神的帮助,让我解除了一直以来的疑惑,感谢前辈们的帮助。

http://blog.csdn.net/luoyanglizi/article/details/51980630





原创粉丝点击