zoukankan      html  css  js  c++  java
  • 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

  • 相关阅读:
    在程序中向水晶报表传参数,以及在程序中指定报表源
    运行Web程序时提示无法使用调试
    TreeView控件节点重命名后没有进入beginEdit的解决方案
    网络负载平衡(转)
    纵横表转交叉表
    重绘datagrid,包括强迫显示某行
    datagrid添加事件
    我的页面模板算法
    C++函数重载
    关于string.empty 与 "" 内存分配
  • 原文地址:https://www.cnblogs.com/diyishijian/p/7762227.html
Copyright © 2011-2022 走看看