Android Studio实现进程间通信AIDL详细步骤

从事Android开发已经好几个年头了,写博客的想法从没有忘记过,而写博客的行动从没有落实过。实现一个功能第一次踩了很多坑,过段时间实现同样的功能还会踩同样的坑,很大一部分原因是因为没有整理、总结。所以我觉得非常有必要将自己掌握的东西总结出来,提升自己能力的同时也能帮助到别人。废话不多说,本篇将是我的第一篇技术博客(从老土的标题应该就能看出来😂),详细讲解Android studio实现AIDL通信步骤。

概述

Android IPC方式有:Intent附加extras传递信息、通过共享文件方式共享数据、Messenger、AIDL、ContentProvider、Socket。AIDL相比于其他几种IPC方式有很多优点:功能强大,支持一对多并发通信,支持实时通讯,可以调用其他进程的方法,应用场景广泛,但缺点就是使用稍微复杂,不过没有关系,本篇将详细讲解AIDL实现步骤,只要动手实现一次就会显得简单了很多。

举个应用场景的栗子:我的小米、小米商城、小米金融、小米视频都属于MIUI系统的预装应用,它们都是独立的apk,当我们打开我的小米,注册登陆之后,再打开小米其他预装应用比如小米视频,会发现小米视频已经登陆了,这是怎么回事呢?

其实这就涉及到了AIDL,当打开小米视频的时候,小米视频(进程)会去调用我的小米(进程)里的方法判断是否登陆,如果登陆还会获取登陆信息(如用户名,昵称等),这样就实现了小米视频自动登陆了。

说明

  • 本篇重点讲解studio实现AIDL步骤,不涉及到AIDL原理;
  • 本篇通过两个apk实现基于AIDL的IPC,更切合实际应用场景;
  • 本篇先实现服务端AIDLService的编写,再实现客户端AIDLClient的编写;
  • 本篇实现的功能是客户端调用服务端方法获取图书列表和向服务端添加图书;
  • 本篇将会详细讲解每个步骤;
  • 文章末尾附上源码;
  • thanks 《Android第一行代码》、《Android开发艺术探索》;

服务端实现详细步骤

1、新建项目AIDLService

注:服务端的包名是com.glong.aidlservice

2、新建Book.java,实现可序列化,提供get/set方法

Book.java

package com.glong.aidlservice;

import android.os.Parcel;
import android.os.Parcelable;

public class Book implements Parcelable {
    private int bookId;
    private String bookName;

    public Book(){
    }

    public Book(int bookId,String bookName){
        this.bookId = bookId;
        this.bookName = bookName;
    }


    @Override
    public int describeContents() {
        return 0;
    }

    @Override
    public void writeToParcel(Parcel dest, int flags) {
        dest.writeInt(this.bookId);
        dest.writeString(this.bookName);
    }

    protected Book(Parcel in) {
        this.bookId = in.readInt();
        this.bookName = in.readString();
    }

    public static final Parcelable.Creator<Book> CREATOR = new Parcelable.Creator<Book>() {
        @Override
        public Book createFromParcel(Parcel source) {
            return new Book(source);
        }

        @Override
        public Book[] newArray(int size) {
            return new Book[size];
        }
    };

    public int getBookId() {
        return bookId;
    }

    public void setBookId(int bookId) {
        this.bookId = bookId;
    }

    public String getBookName() {
        return bookName;
    }

    public void setBookName(String bookName) {
        this.bookName = bookName;
    }
}

这一步很简单,五秒搞定,还不能五秒搞定的童鞋戳这里

3、新建AIDL Folder

生成的aidl文件夹跟java文件夹属于同一层级

4、在aidl目录下新建IBookManager.aidl

IDE会自动在aidl文件夹下创建com.glong.aidlservice包,并在该包下帮我们生成IBookManager.aidl

我们稍做改动增加两个接口,getBookList()获取图书列表 addBook(in Book book)向服务端添加一本书

IBookManager.aidl

package com.glong.aidlservice;
import com.glong.aidlservice.Book;
// Declare any non-default types here with import statements

interface IBookManager {

    /*
     * 向客户端提供获取图书列表接口
     */
    List<Book> getBookList();

    /*
     * 向客户端提供添加图书接口
     */
    void addBook(in Book book);
}

注意:

  • 第二行显式import了Book类,AIDL中所有Parcelable对象要显式的import进来,而此处我们接口中使用到了自定义的Parcelable对象Book类型;

  • 在AIDL文件中用到了自定义的Parcelable对象时,必须新建一个和该对象同名的.aidl文件,所以接下来我们还需新建Book.aidl

    以上两点如果没有import或者import包名有误或者没有创建Parcelable对象对应的.aidl文件,项目就不能够编译通过,会报"Process 'command 'D:\SDK\build-tools\27.0.3\aidl.exe'' finished with non-zero exit value 1"错误,找不到对应的对象;

  • addBook()方法形参前的in表示方向。AIDL中除基本数据类型,其他类型的参数必须标上方向:in、out或者inoutin表示输入性参数,out表示输出型参数,inout表示输入输出型参数;

5、在aidl目录下的com.glong.aidlservice包下新建Book.aidl

注意如果直接New → ADIL→ AIDL File 然后输入Book,IDE会提示"interface Name must be unique",我们使用如下步骤创建Book.aidl;




Book.aidl

package com.glong.aidlservice;

parcelable Book;

6、Build编译项目

编译完成后将项目结构模式切换成Project,在build文件夹下可以看到IDE帮我们生成了IBookManager.java由于本篇重点将步骤,这个文件不多做讲解,项目结构重新切回到Android模式

7、实现远程Service

在java文件夹的com.glong.aidlservice包下新建BookManagerService.java

BookManagerService.java

package com.glong.aidlservice;

import android.app.Service;
import android.content.ContentValues;
import android.content.Intent;
import android.database.Cursor;
import android.database.sqlite.SQLiteDatabase;
import android.os.Binder;
import android.os.IBinder;
import android.os.RemoteException;
import android.support.annotation.Nullable;
import android.util.Log;

import com.glong.aidlservice.database.MyDatabaseHelper;

import java.util.ArrayList;
import java.util.List;

public class BookManagerService extends Service {
    private static final String TAG = "guo.BookManagerService";

    private SQLiteDatabase database;
    private List<Book> mBookList = new ArrayList<>();

    private Binder mBinder = new IBookManager.Stub() {

        @Override
        public List<Book> getBookList() throws RemoteException {
            return mBookList;
        }

        @Override
        public void addBook(Book book) throws RemoteException {
            addBookIntoDB(book);
        }
    };

    @Nullable
    @Override
    public IBinder onBind(Intent intent) {
        return mBinder;
    }

    @Override
    public void onCreate() {
        super.onCreate();
        Log.d(TAG, "service onCreate");

        queryBooksFromDB();
    }

    /**
     * 从数据库中获取Book列表,数据库操作{@link MyDatabaseHelper}
     */
    private void queryBooksFromDB() {
        Log.i(TAG, "service starting query Book from db.");
        MyDatabaseHelper dbHelper = new MyDatabaseHelper(this, "BookStore.db", null, 1);
        database = dbHelper.getReadableDatabase();
        Cursor cursor = database.query("Book", null, null, null, null, null, null);
        if (cursor.moveToFirst()) {
            do {
                int bookId = cursor.getInt(cursor.getColumnIndex("bookId"));
                String bookName = cursor.getString(cursor.getColumnIndex("bookName"));
                Book book = new Book(bookId, bookName);
                mBookList.add(book);
            } while (cursor.moveToNext());
        }
        cursor.close();
    }

    private void addBookIntoDB(Book book) {
        Log.i(TAG, "service starting add Book to db.");
        ContentValues values = new ContentValues();
        values.put("bookId", book.getBookId());
        values.put("bookName", book.getBookName());
        database.insert("Book", null, values);
    }
}

服务端的Book列表存储在数据库中,当客户端远程连接服务端时,BookManagerService服务开启,并且在onCreate()中查找数据库初始化mBookList图书列表,下面再附上MyDatabaseHelper.java

package com.glong.aidlservice.database;

import android.content.Context;
import android.database.sqlite.SQLiteDatabase;
import android.database.sqlite.SQLiteOpenHelper;
import android.util.Log;

public class MyDatabaseHelper extends SQLiteOpenHelper {

    private static final String TAG = "long.MyDatabaseHelper";

    private static final String CREATE_BOOK = "create table Book (" +
            "id integer primary key autoincrement," +
            "bookId integer," +
            "bookName text)";

    public MyDatabaseHelper(Context context, String name, SQLiteDatabase.CursorFactory factory, int version) {
        super(context, name, factory, version);
    }

    @Override
    public void onCreate(SQLiteDatabase sqLiteDatabase) {
        sqLiteDatabase.execSQL(CREATE_BOOK);
        Log.i(TAG, "Book database create succeeded!");
    }

    @Override
    public void onUpgrade(SQLiteDatabase sqLiteDatabase, int i, int i1) {

    }
}

代码很简单,就是简单的创建数据库。
最后不要忘记在AndroidMenifest.xml中配置该服务,并且指定启动的Action,以便客户端通过此Action启动远程服

    <service android:name=".BookManagerService">
        <intent-filter>
            <action android:name="com.glong.aidlservice.action.BookManagerService" />
        </intent-filter>
    </service>

8、给数据库添加数据,方面后面我们后面验证客户端是否请求到了服务端的数据

修改MainActivity.java代码:

package com.glong.aidlservice;

import android.content.ContentValues;
import android.database.sqlite.SQLiteDatabase;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;

import com.glong.aidlservice.database.MyDatabaseHelper;

public class MainActivity extends AppCompatActivity {

    private SQLiteDatabase database;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        MyDatabaseHelper dbHelper = new MyDatabaseHelper(this, "BookStore.db", null, 1);
        database = dbHelper.getReadableDatabase();

        //如果数据库为空就向数据库添加几个数据
        if (!database.query("Book", null, null, null, null, null, null).moveToFirst()) {
            addBooksToDB();
        }
    }

    private void addBooksToDB() {
        ContentValues values = new ContentValues();
        values.put("bookId", 0);
        values.put("bookName", "玉女真经");
        database.insert("Book", null, values);
        values.clear();

        values.put("bookId", 1);
        values.put("bookName", "辟邪剑法");
        database.insert("Book", null, values);
        values.clear();

        values.put("bookId", 2);
        values.put("bookName", "如来神掌");
        database.insert("Book", null, values);
        values.clear();

        values.put("bookId", 3);
        values.put("bookName", "降龙十八掌");
        database.insert("Book", null, values);
        values.clear();

        values.put("bookId", 4);
        values.put("bookName", "九阴真经");
        database.insert("Book", null, values);
        values.clear();

        values.put("bookId", 5);
        values.put("bookName", "九阳神功");
        database.insert("Book", null, values);
        values.clear();

        values.put("bookId", 6);
        values.put("bookName", "打狗棒法");
        database.insert("Book", null, values);
        values.clear();

        values.put("bookId", 7);
        values.put("bookName", "乾坤大挪移");
        database.insert("Book", null, values);
        values.clear();

        values.put("bookId", 8);
        values.put("bookName", "易筋经");
        database.insert("Book", null, values);
        values.clear();

        values.put("bookId", 9);
        values.put("bookName", "六脉神剑");
        database.insert("Book", null, values);
        values.clear();

        values.put("bookId", 10);
        values.put("bookName", "黯然销魂掌");
        database.insert("Book", null, values);
        values.clear();

        values.put("bookId", 11);
        values.put("bookName", "九阴白骨爪");
        database.insert("Book", null, values);
        values.clear();

        values.put("bookId", 12);
        values.put("bookName", "独孤九剑");
        database.insert("Book", null, values);
    }
}

同样代码很简单,先在onCreate()方法中获取database对象,判断如果database没有数据,则调用addBooksToDB()方法添加十几本书(加上这个判断就不会出现每进入一次服务端应用,数据库就多一组重复的数据)。

到此,服务端已经写完了,下面开始写客户端。

客户端实现详细步骤

客户端的实现前三步和服务端一样,这里依旧贴上步骤

1、新建项目AIDLClient

注:客户端的包名是com.glong.aidlclient

2、新建Book.java,实现可序列化,提供get/set方法

和服务端Book.java保持一致

Book.java

package com.glong.aidlclient;

import android.os.Parcel;
import android.os.Parcelable;

public class Book implements Parcelable {
    private int bookId;
    private String bookName;

    public Book() {
    }

    public Book(int bookId, String bookName) {
        this.bookId = bookId;
        this.bookName = bookName;
    }

    @Override
    public int describeContents() {
        return 0;
    }

    @Override
    public void writeToParcel(Parcel dest, int flags) {
        dest.writeInt(this.bookId);
        dest.writeString(this.bookName);
    }

    protected Book(Parcel in) {
        this.bookId = in.readInt();
        this.bookName = in.readString();
    }

    public static final Parcelable.Creator<Book> CREATOR = new Parcelable.Creator<Book>() {
        @Override
        public Book createFromParcel(Parcel source) {
            return new Book(source);
        }

        @Override
        public Book[] newArray(int size) {
            return new Book[size];
        }
    };

    public int getBookId() {
        return bookId;
    }

    public void setBookId(int bookId) {
        this.bookId = bookId;
    }

    public String getBookName() {
        return bookName;
    }

    public void setBookName(String bookName) {
        this.bookName = bookName;
    }
}

3、新建AIDL Folder

4、在aidl目录下新建IBookManager.aidl

这里需要注意两点:

  • 客户端的IBookManager.aidl必须跟服务端的IBookMananger.aidl包名相同,否则客户端运行连接服务端时会报SecturityException;

  • 客户端的IBookManager.aidl必须跟服务端的IBookManager.aidl拥有相同的方法和方法数,否则在通过AIDL进行IPC时会出现乱码等现象;

    总之,客户端和服务端的IBookManager.aidl保持一致(除import的Parcelable对象部分),包括包名和aidl文件夹一下的路径,所以这一步的创建IBookManager.aidl和服务端的创建稍有不同,aidl文件夹下 → New → Package

然后输入服务端IBookManager.aidl的包名,点击OK

接着在该包下创建IBookMananger.aidl,New → FIle

接下来实现IBookManager.aidl,或者copy服务端IBookManager.aidl代码,修改下import的包即可(import的包和当前工程的包对应

IBookManager.aidl

package com.glong.aidlservice;

import com.glong.aidlclient.Book;
interface IBookManager{

    /*
     * 获取Book列表
     */
    List<Book> getBookList();

    /*
     *添加Book
     */
     List<Book> addBook(in Book book);

}

可以看到IBookManager.aidl文件包名和服务端包名一样
值得注意的是import com.glong.aidlclient.Book和服务端的略有区别,import的包名和当前工程包名须对应,所以接下来创建的Book.aidlcom.glong.aidlclient包下

5、在aidl目录下新建com.glong.aidlclient包,并在该包下新建Book.aidl

先新建 Package com.glong.aidlclient

新建com.glong.aidlclient

接着在com.glong.aidlclient包下新建Book.aidl,新建之后工程目录是这样的



Book.aidl

package com.glong.aidlclient;

parcelable Book;

6、Build编译项目

这一步跟服务端一样,不过多赘述

7、客户端实现

  • 首先绑定远程服务

  • 绑定成功后将服务端返回的Binder对象转换成AIDL接口

  • 通过这个AIDL接口调用远程方法

MainActivity.java

package com.glong.aidlclient;

import android.content.ComponentName;
import android.content.Context;
import android.content.Intent;
import android.content.ServiceConnection;
import android.os.IBinder;
import android.os.RemoteException;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.util.Log;
import android.widget.Toast;

import com.glong.aidlservice.IBookManager;

import java.util.List;

public class MainActivity extends AppCompatActivity {

    private static final String TAG = "long.MainActivity";
    private static final String SERVICE_ACTION = "com.glong.aidlservice.action.BookManagerService";
    private static final String SERVICE_PACKAGE = "com.glong.aidlservice";

    private IBookManager mBookManager;

    ServiceConnection connection = new ServiceConnection() {
        @Override
        public void onServiceConnected(ComponentName componentName, IBinder iBinder) {
            mBookManager = IBookManager.Stub.asInterface(iBinder);
            try {
                List<Book> books = mBookManager.getBookList();
                if (books == null || books.size() == 0) return;
                for (int i = 0; i < books.size(); i++) {
                    Book book = books.get(i);
                    Log.i(TAG, i + ".[bookId:" + book.getBookId() + ",bookName:" + book.getBookName() + "]");
                }
            } catch (RemoteException e) {
                e.printStackTrace();
                Toast.makeText(MainActivity.this, "获取Book列表失败", Toast.LENGTH_SHORT).show();
            }
        }

        @Override
        public void onServiceDisconnected(ComponentName componentName) {

        }
    };

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        bindService();
    }

    /**
     * 连接远程服务
     */
    private void bindService() {
        Intent intent = new Intent();
        intent.setPackage(SERVICE_PACKAGE);
        intent.setAction(SERVICE_ACTION);
        bindService(intent, connection, Context.BIND_AUTO_CREATE);
    }
}

代码依旧很简单,这里使用隐式的方式绑定远程服务,在onServiceConnected()回调方法中获取服务端Book列表并打印出来。

到这里,服务端的实现也完成了,以上是整个AIDL实现步骤,后面内容属于验证和扩展可以不用关心

验证

  • run AIDLService工程
  • run AIDLClient工程
  • 查看日志
验证

我们调用远程服务的接口成功了!

扩展

  • 客户端调用远程服务getBookList()接口返回数据后,展示数据;
  • 客户端调用addBook(Book book)接口增加图书

修改activity_main.xml

<?xml version="1.0" encoding="utf-8"?>
<FrameLayout 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"
    tools:context=".MainActivity">


    <Button
        android:id="@+id/btn_book_list"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center"
        android:text="获取Book列表"
        android:background="#603300cc"
        android:textSize="22sp" />

    <Button
        android:id="@+id/btn_book_add"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="right|bottom"
        android:background="#903300cc"
        android:text="添加图书"
        android:textSize="22sp"
        android:visibility="gone" />

    <android.support.v7.widget.RecyclerView
        android:id="@+id/recycler_view_book_list"
        android:layout_width="match_parent"
        android:layout_height="match_parent"
        android:visibility="gone" />


</FrameLayout>

两个Button分别对应获取图书列表功能和添加图书功能,RecyclerView展示图书列表

1、实现展示图书列表功能

MyAdapter.java

package com.glong.aidlclient;

import android.graphics.Color;
import android.support.annotation.NonNull;
import android.support.v7.widget.RecyclerView;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.TextView;

import java.util.List;

public class MyAdapter extends RecyclerView.Adapter<MyAdapter.ViewHolder> {

    private List<Book> mBooks;

    public MyAdapter(List<Book> books) {
        mBooks = books;
    }

    @NonNull
    @Override
    public ViewHolder onCreateViewHolder(@NonNull ViewGroup viewGroup, int i) {
        View view = LayoutInflater.from(viewGroup.getContext()).inflate(R.layout.item_view, viewGroup, false);
        //这里给view设置一下间隙
        RecyclerView.LayoutParams lp = (RecyclerView.LayoutParams) view.getLayoutParams();
        lp.setMargins(0, 10, 0, 10);
        view.setBackgroundColor(Color.YELLOW);
        return new ViewHolder(view);
    }

    @Override
    public void onBindViewHolder(@NonNull ViewHolder viewHolder, int i) {
        Book book = mBooks.get(i);
        viewHolder.bookIdTextView.setText("\t\t" + (i + 1) + "." + "《" + book.getBookName() + "》");
        viewHolder.bookNameTextView.setText("ID:" + String.format("%08d", book.getBookId()));
    }

    @Override
    public int getItemCount() {
        return mBooks.size();
    }

    static class ViewHolder extends RecyclerView.ViewHolder {

        TextView bookIdTextView;
        TextView bookNameTextView;

        public ViewHolder(@NonNull View itemView) {
            super(itemView);
            bookIdTextView = itemView.findViewById(R.id.tv_book_id);
            bookNameTextView = itemView.findViewById(R.id.tv_book_name);
        }
    }
}

item_view.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:orientation="horizontal">

    <TextView
        android:id="@+id/tv_book_id"
        android:layout_width="0dp"
        android:layout_height="wrap_content"
        android:layout_weight="1"
        android:gravity="center_vertical"
        android:minHeight="?android:attr/listPreferredItemHeightSmall"
        android:paddingEnd="?android:attr/listPreferredItemPaddingEnd"
        android:paddingStart="?android:attr/listPreferredItemPaddingStart"
        android:textAppearance="?android:attr/textAppearanceListItemSmall"
        tools:ignore="RtlCompat" />

    <TextView
        android:id="@+id/tv_book_name"
        android:layout_width="0dp"
        android:layout_height="wrap_content"
        android:layout_weight="1"
        android:gravity="center_vertical"
        android:minHeight="?android:attr/listPreferredItemHeightSmall"
        android:paddingEnd="?android:attr/listPreferredItemPaddingEnd"
        android:paddingStart="?android:attr/listPreferredItemPaddingStart"
        android:textAppearance="?android:attr/textAppearanceListItemSmall" />

</LinearLayout>

修改MainAcitivty.java

package com.glong.aidlclient;

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.support.v7.widget.DividerItemDecoration;
import android.support.v7.widget.LinearLayoutManager;
import android.support.v7.widget.RecyclerView;
import android.util.Log;
import android.view.View;
import android.widget.Button;
import android.widget.Toast;

import com.glong.aidlservice.IBookManager;

import java.util.ArrayList;
import java.util.List;

public class MainActivity extends AppCompatActivity implements View.OnClickListener {

    private static final String TAG = "long.MainActivity";
    private static final String SERVICE_ACTION = "com.glong.aidlservice.action.BookManagerService";
    private static final String SERVICE_PACKAGE = "com.glong.aidlservice";

    private IBookManager mBookManager;

    private Button mBtnRequestBooks;
    private Button mBtnAddBook;
    private RecyclerView mRecyclerView;

    private List<Book> mBooks = new ArrayList<>();
    private MyAdapter mAdapter;

    private AddBookDialog mDialog;

    ServiceConnection connection = new ServiceConnection() {
        @Override
        public void onServiceConnected(ComponentName componentName, IBinder iBinder) {
            mBookManager = IBookManager.Stub.asInterface(iBinder);
            try {
                List<Book> list = mBookManager.getBookList();
                if (list != null && list.size() > 0) {
                    mBooks.addAll(list);
                }
                mRecyclerView.setVisibility(View.VISIBLE);
                mBtnRequestBooks.setVisibility(View.GONE);
                mBtnAddBook.setVisibility(View.VISIBLE);
                mAdapter.notifyDataSetChanged();
            } catch (RemoteException e) {
                e.printStackTrace();
                Toast.makeText(MainActivity.this, "获取Book列表失败", Toast.LENGTH_SHORT).show();
            }
        }

        @Override
        public void onServiceDisconnected(ComponentName componentName) {
            Toast.makeText(MainActivity.this, "onServiceDisconnected!", Toast.LENGTH_SHORT).show();
        }
    };

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        initViews();
        initRecyclerView();
    }

    private void initViews() {
        mBtnAddBook = findViewById(R.id.btn_book_add);
        mBtnRequestBooks = findViewById(R.id.btn_book_list);

        mBtnRequestBooks.setOnClickListener(this);
        mBtnAddBook.setOnClickListener(this);
    }

    private void initRecyclerView() {
        mRecyclerView = findViewById(R.id.recycler_view_book_list);
        LinearLayoutManager linearLayoutManager = new LinearLayoutManager(this);
        mRecyclerView.setLayoutManager(linearLayoutManager);
        mRecyclerView.addItemDecoration(new DividerItemDecoration(this, DividerItemDecoration.VERTICAL));
        mAdapter = new MyAdapter(mBooks);
        mRecyclerView.setAdapter(mAdapter);
    }

    /**
     * 连接远程服务
     */
    private void bindService() {
        Intent intent = new Intent();
        intent.setPackage(SERVICE_PACKAGE);
        intent.setAction(SERVICE_ACTION);
        bindService(intent, connection, Context.BIND_AUTO_CREATE);
    }

    @Override
    public void onClick(View view) {
        switch (view.getId()) {
            case R.id.btn_book_list:
                requestBookList();
                break;
            case R.id.btn_book_add:
                showAddBookDialog();
                break;
            default:
                break;
        }
    }

    /**
     * show 添加图书的Dialog
     */
    private void showAddBookDialog() {
        if (mDialog == null)
            mDialog = new AddBookDialog(this, new AddBookDialog.OnDialogDismissCallback() {
                @Override
                public void onOkButtonPressed(int bookId, String bookName) {
                    Log.d(TAG, "Ok button pressed!");
                    addBook(bookId, bookName);
                }

                @Override
                public void onCancelButtonPressed() {

                }
            });
        mDialog.show();
    }

    /**
     * 添加Book{@link Book}
     */
    private void addBook(int bookId, String bookName) {
        Log.i(TAG, "client add Book starting...id:" + bookId + " ,bookName:" + bookName);
        if (mBookManager == null) {
            Log.i(TAG, "mBookManager == null!!");
            return;
        }
        try {
            Book book = new Book(bookId, bookName);
            mBookManager.addBook(book);

            mBooks.add(book);
            mAdapter.notifyDataSetChanged();
            Toast.makeText(this, "Add succeed", Toast.LENGTH_SHORT).show();
        } catch (RemoteException e) {
            e.printStackTrace();
            Toast.makeText(this, "添加失败", Toast.LENGTH_SHORT).show();
        }
    }

    /**
     * 获取Book列表
     */
    private void requestBookList() {
        bindService();
    }
}

2、实现向服务端添加图书功能

功能很简单,点击“添加图书”按钮后弹出dialog,输入bookId和bookName,点击“确定”

AddBookDialog.java

package com.glong.aidlclient;

import android.app.Activity;
import android.app.Dialog;
import android.content.Context;
import android.os.Bundle;
import android.support.annotation.NonNull;
import android.text.TextUtils;
import android.view.View;
import android.widget.AutoCompleteTextView;
import android.widget.Button;
import android.widget.EditText;
import android.widget.Toast;

public class AddBookDialog extends Dialog implements View.OnClickListener {

    private OnDialogDismissCallback callback;
    private Button mBtnOk;
    private Button mBtnCancel;
    private EditText mBookIdTextView;
    private AutoCompleteTextView mBookNameTextView;

    public AddBookDialog(@NonNull Context context, OnDialogDismissCallback callback) {
        this(context, R.style.dialog, callback);
    }

    public AddBookDialog(@NonNull Context context, int themeResId, OnDialogDismissCallback callback) {
        super(context, themeResId);
        this.callback = callback;
        setOwnerActivity((Activity) context);
    }

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.dialog_layout);
        setCanceledOnTouchOutside(false);

        mBtnOk = findViewById(R.id.button_ok);
        mBtnCancel = findViewById(R.id.button_cancel);
        mBookIdTextView = findViewById(R.id.edit_book_id);
        mBookNameTextView = findViewById(R.id.edit_book_name);

        mBtnOk.setOnClickListener(this);
        mBtnCancel.setOnClickListener(this);
    }

    @Override
    public void show() {
        super.show();
        mBookIdTextView.setText("");
        mBookNameTextView.setText("");
    }

    @Override
    public void onClick(View view) {
        switch (view.getId()) {
            case R.id.button_ok:
                pressedOkBtn();
                break;
            case R.id.button_cancel:
                pressedCancelBtn();
                break;
            default:
                break;
        }
    }

    private void pressedCancelBtn() {
        dismiss();
    }

    private void pressedOkBtn() {
        String idString = mBookIdTextView.getText().toString();
        String name = mBookNameTextView.getText().toString();

        if (idString.isEmpty()) {
            Toast.makeText(getContext(), "ID不能为空", Toast.LENGTH_SHORT).show();
            return;
        }
        if (!idString.matches("\\d+")) {
            Toast.makeText(getContext(), "ID格式不正确", Toast.LENGTH_SHORT).show();
            return;
        }
        if (TextUtils.isEmpty(name)) {
            Toast.makeText(getContext(), "书名不能为空", Toast.LENGTH_SHORT).show();
            return;
        }
        dismiss();
        if (callback == null) return;
        int id = Integer.parseInt(idString);
        callback.onOkButtonPressed(id, name);
    }

    public interface OnDialogDismissCallback {
        /**
         * 通过点击确定按钮返回的
         *
         * @param bookId
         * @param bookName
         */
        void onOkButtonPressed(int bookId, String bookName);

        /**
         * 通过点击取消按钮返回的
         */
        void onCancelButtonPressed();
    }
}

Dialog的布局文件dialog_layout.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:background="#FFCC99"
    android:orientation="vertical">

    <TextView
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_gravity="center_horizontal"
        android:layout_marginTop="20dp"
        android:text="添加图书"
        android:textColor="#60000000" />

    <LinearLayout
        android:layout_width="300dp"
        android:layout_height="wrap_content"
        android:layout_gravity="center_horizontal"
        android:layout_marginTop="20dp"
        android:orientation="horizontal"
        android:paddingLeft="10dp"
        android:paddingRight="10dp">

        <TextView
            android:layout_width="80dp"
            android:layout_height="wrap_content"
            android:lines="1"
            android:text="Book ID:"
            android:textColor="#60000000" />

        <EditText
            android:id="@+id/edit_book_id"
            android:layout_width="0dp"
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:hint="请输入ID"
            android:inputType="number" />
    </LinearLayout>

    <LinearLayout
        android:layout_width="300dp"
        android:layout_height="wrap_content"
        android:layout_gravity="center_horizontal"
        android:layout_marginTop="20dp"
        android:orientation="horizontal"
        android:paddingLeft="10dp"
        android:paddingRight="10dp">

        <TextView
            android:layout_width="80dp"
            android:layout_height="wrap_content"
            android:lines="1"
            android:text="Book Name:"
            android:textColor="#60000000" />

        <AutoCompleteTextView
            android:id="@+id/edit_book_name"
            android:layout_width="0dp"
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:hint="请输入Name"
            android:inputType="text" />

    </LinearLayout>

    <FrameLayout
        android:layout_width="300dp"
        android:layout_height="wrap_content"
        android:layout_gravity="center_horizontal"
        android:layout_marginBottom="20dp"
        android:layout_marginTop="20dp">

        <Button
            android:id="@+id/button_ok"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_gravity="left|center_vertical"
            android:layout_marginLeft="50dp"
            android:text="确定" />

        <Button
            android:id="@+id/button_cancel"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
            android:layout_gravity="right|center_vertical"
            android:layout_marginRight="50dp"
            android:text="取消" />
    </FrameLayout>
</LinearLayout>

现在重新run AIDLClient工程效果如下:


c8b51ab6-f17a-401a-8541-dff13d5c80fb.gif


源码下载

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

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 170,569评论 25 707
  • Android跨进程通信IPC整体内容如下 1、Android跨进程通信IPC之1——Linux基础2、Andro...
    隔壁老李头阅读 10,564评论 13 43
  • 本篇包含进程间通信——AIDL所涉及到的知识的自我总结(内容详细) 通过前段时间对AIDL的学习以及最近一些资料的...
    arvinljw阅读 2,939评论 0 17
  • 慈母手中线,游子身上衣,临行密密缝,意恐迟迟归。今天是母亲节,想讲三件小事. 1.之所以把这篇小诗放到篇首,是因为...
    wenju_song阅读 191评论 0 2
  • 交通 济南出发机票来回往返大概900元左右。700元左右的机票到重庆大概凌晨,时间不太好。看自行考虑吧。 重庆江北...
    _siximie阅读 220评论 0 0