Android进程通信IPC

IPC也看过很多次,每次看完后记会忘记,这次自己给它写下来加深印象

IPC

进程间通信(Inter-Process Communication),简称IPC,就是指进程与进程之间进行通信

基于Binder的AIDL和Messager的同一个应用的通信

步骤一

首先写个最基本的就是重写IBinder,增强服务端的的功能 看代码:

//IpcService.java文件
public class IpcService extends Service {
    private IBinder mIBinder;

    @Override
    public void onCreate() {
        super.onCreate();
        mIBinder = new MyBinder();
    }

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

    public class MyBinder extends Binder {
        public void print() {
            Log.d("PHN", "服务Service,执行任务");
        }
    }

}

注意要在清单文件中注册服务

public class MainActivity extends AppCompatActivity {

    private IpcService.MyBinder myBinder;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        //自动绑定服务
        Intent intent = new Intent(this, IpcService.class);
        MyServiceConnection serviceConnection = new MyServiceConnection();
        bindService(intent, serviceConnection, BIND_AUTO_CREATE);

        findViewById(R.id.bt).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if (null != myBinder) {
                    myBinder.print();
                }
            }
        });
    }

    private class MyServiceConnection implements ServiceConnection {
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            myBinder = (IpcService.MyBinder) service;
        }

        @Override
        public void onServiceDisconnected(ComponentName name) {

        }
    }
}
loginfo_01.png

执行点击服务端会打印信息,说明这个活动和服务进行通信是成功的!

步骤二

我们用Messager来做刚才的功能
1.首先在Service里面创建一个Hander用来接受消息:

private final static Handler mHandler = new Handler() {
    @Override
    public void handleMessage(Message msg) {
       Log.d("PHN", "服务Service,执行任务");
    }
};

2.在Service里面创建一个Messager,并把Handler放入其中

private final static Messenger mMessenger = new Messenger(mHandler);

3.重写onbind方法,返回Messager里面的Binder

public IBinder onBind(Intent intent) {
    return mMessenger.getBinder();
}

贴出完整的服务代码


public class IpcService extends Service {

    @Override
    public void onCreate() {
        super.onCreate();
    }

    @Nullable
    @Override
    public IBinder onBind(Intent intent) {
        return mMessenger.getBinder();
    }


    private final static Handler mHandler = new Handler() {
        @Override
        public void handleMessage(Message msg) {
            Log.d("PHN", "服务Service,执行任务");
        }
    };
    private final static Messenger mMessenger = new Messenger(mHandler);

}

再贴出活动里的代码

public class MainActivity extends AppCompatActivity {

    private Messenger messenger;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        //自动绑定服务
        Intent intent = new Intent(this, IpcService.class);
        MyServiceConnection serviceConnection = new MyServiceConnection();
        bindService(intent, serviceConnection, BIND_AUTO_CREATE);

        findViewById(R.id.bt).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if (null != messenger) {
                    Message message = Message.obtain();
                    try {
                        messenger.send(message);
                    } catch (RemoteException e) {
                        e.printStackTrace();
                    }
                }
            }
        });
    }

    private class MyServiceConnection implements ServiceConnection {
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            messenger = new Messenger(service);
        }

        @Override
        public void onServiceDisconnected(ComponentName name) {

        }
    }
}

执行看看结果

loginfo_02.png

果然可以得到同样的效果!

步骤三

上面我们可能看到只能是活动让服务执行任务,那么怎样才能让服务端让活动也执行任务呢,我们先对服务器端的代码进行修改,首先修改Service的Handler
看下面服务端代码处理:

public class IpcService extends Service {

    @Override
    public void onCreate() {
        super.onCreate();
    }

    @Nullable
    @Override
    public IBinder onBind(Intent intent) {
        return mMessenger.getBinder();
    }


    private final static Handler mHandler = new Handler() {
        @Override
        public void handleMessage(Message msg) {
            Log.d("PHN", "服务Service,执行任务");
            //获取Messager
            Messenger messenger = msg.replyTo;
            //创建消息
            Message msg_reply = Message.obtain();
            try {
                //发送
                messenger.send(msg_reply);
            } catch (RemoteException e) {
                e.printStackTrace();
            }
        }
    };
    private final static Messenger mMessenger = new Messenger(mHandler);

}

接着我们在活动端也增加一个Handler和Messager来处理消息,贴出代码

public class MainActivity extends AppCompatActivity {

    private Messenger messenger;


    private final static Handler mHandler = new Handler() {
        @Override
        public void handleMessage(Message msg) {
            Log.d("PHN", "活动Acitivity,执行任务");
        }
    };
    private final static Messenger mReplyMessager = new Messenger(mHandler);


    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        //自动绑定服务
        Intent intent = new Intent(this, IpcService.class);
        MyServiceConnection serviceConnection = new MyServiceConnection();
        bindService(intent, serviceConnection, BIND_AUTO_CREATE);

        findViewById(R.id.bt).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                if (null != messenger) {
                    Message message = Message.obtain();
                    //通过msg把客户端的Messager传送到服务器端(关键代码)
                    message.replyTo =mReplyMessager;
                    try {
                        messenger.send(message);
                    } catch (RemoteException e) {
                        e.printStackTrace();
                    }
                }
            }
        });
    }

    private class MyServiceConnection implements ServiceConnection {
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            messenger = new Messenger(service);
        }

        @Override
        public void onServiceDisconnected(ComponentName name) {

        }
    }
}

还有一个比较关键的地方,就是要在客户端发送消息的时候把客户端的Messager通过消息传送到服务器端
(msg.replyTo =mReplyMessager)

来执行看看结果:

loginfo_03.png

果然可以!!

步骤四

上面的通信是基于本应用里的,下面我们来看看不同的两个应用之间的通信AIDL,我是参照任玉刚的开发艺术里写例子,
1.首先那就是新建AIDL文件,很简单:

screenshot_01.png

那我就直接创建两个aidl文件Book.aidl,BookManager.aidl

screenshot_02.png

下面直接看文件的具体内容,我贴上来:

//Book.java
public class Book implements Parcelable{
    protected 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];
        }
    };

    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;
    }

    private String name;

    private int price;

    public Book() {}

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

    @Override
    public void writeToParcel(Parcel dest, int flags) {
        dest.writeString(name);
        dest.writeInt(price);
    }
    /**
     * 参数是一个Parcel,用它来存储与传输数据
     * @param dest
     */
    public void readFromParcel(Parcel dest) {
        //注意,此处的读值顺序应当是和writeToParcel()方法中一致的
        name = dest.readString();
        price = dest.readInt();
    }

    @Override
    public String toString() {
        return "Book{" +
                "name='" + name + '\'' +
                ", price=" + price +
                '}';
    }
}
// Booka.aidl
package com.hanny.aidldemo;

// Declare any non-default types here with import statements
parcelable Book;

// BookManager.aidl
package com.hanny.aidldemo;
import com.hanny.aidldemo.Book;
// Declare any non-default types here with import statements

interface BookManager {

    //所有的返回值前都不需要加任何东西,不管是什么数据类型
      List getBooks();

      //传参时除了Java基本类型以及String,CharSequence之外的类型
      //都需要在前面加上定向tag,具体加什么量需而定
      void addBook(in Book book);
}

文件搞好了,先编译下,看看会不会错,应该会错。

loginfo_03.png

这是因为把Book.java文件放到了aidl下,所以要在build.gradle 的android{ }的闭包下加入

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

不会错了,那就来写AIDL的service文件:

public class AIDLService extends Service {

    //包含Book对象的list
    private List mBooks = new ArrayList<>();
    private BookManager.Stub bookManager = new BookManager.Stub() {
        @Override
        public List getBooks() throws RemoteException {
            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) {
                    book = new Book();
                }
                book.setPrice(23);
                if (!mBooks.contains(book)) {
                    mBooks.add(book);
                }
                //打印mBooks列表,观察客户端传过来的值
                Log.e("PHN", "增加后的图书 : " + mBooks.toString());
            }
        }
    };

    @Override
    public void onCreate() {
        super.onCreate();
        Book book = new Book();
        book.setName("Android开发艺术探索");
        book.setPrice(28);
        mBooks.add(book);
    }

    @Nullable
    @Override
    public IBinder onBind(Intent intent) {

        return bookManager;
    }
}

服务端写好了,那就新建一个工程,然后把服务端的aidl下的文件全部拷贝过来~放在main下级目录!

screenshot_03.png

接着在活动里用隐式意图开启服务!这样的话就要在服务端的清单文件里给服务配上action

   <service android:name=".AIDLService">
            <intent-filter>
                <action android:name="action.ipc"/>
            </intent-filter>
        </service>

看看客户端活动里的代码:

public class MainActivity extends AppCompatActivity {

    //标志当前与服务端连接状况的布尔值,false为未连接,true为连接中
    private boolean mBound = false;
    private BookManager bookManager;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        findViewById(R.id.bt_get).setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                addBook();
            }
        });
    }

    private void addBook() {
        if (!mBound) {
            attemptToBindService();
            Toast.makeText(this, "当前与服务端处于未连接状态,正在尝试重连,请稍后再试", Toast.LENGTH_SHORT).show();
            return;
        }
        if (bookManager == null) return;

        Book book = new Book();
        book.setName("APP研发录In");
        book.setPrice(30);
        try {
            bookManager.addBook(book);
            Log.e(getLocalClassName(), book.toString());
        } catch (RemoteException e) {
            e.printStackTrace();
        }
    }

    //开启服务
    private void attemptToBindService() {
        Intent intent = new Intent();
        intent.setAction("action.ipc");
        intent.setPackage("com.hanny.aidldemo");
        bindService(intent, mServiceConnection, BIND_AUTO_CREATE);
    }

    private ServiceConnection mServiceConnection = new ServiceConnection() {
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            bookManager = BookManager.Stub.asInterface(service);
            mBound = true;
            if (bookManager != null) {
                try {
                    List books = bookManager.getBooks();
                    Log.d("PHN", "初始化得到的书: "+books.toString());
                } catch (RemoteException e) {
                    e.printStackTrace();
                }
            }
        }

        @Override
        public void onServiceDisconnected(ComponentName name) {
            mBound = false;
        }
    };


    @Override
    protected void onStop() {
        super.onStop();
        if (mBound) {
            unbindService(mServiceConnection);
            mBound = false;
        }
    }

}

下面先运行服务端apk ,然后运行客服务apk然后点击增加图书,看看客户端和服务端的打印结果:

Paste_Image.png

反正就是这么简单,动手实践加深印象,并没有什么难的地方,

注意地方

AIDL 发生异常的原因 Android java.lang.SecurityException: Binder invocation to an incorrect interface

解决方法如下:
在使用上请注意,服务端与客户端都要有相同的接口(使用到的),这里的“相同”是指完全相同,包括包名,也就是说要在不同工程下建立相同的包名,这样一来,问题应该迎刃而解了!

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

推荐阅读更多精彩内容