android多进程及AIDL简单使用

正常情况下,一个apk启动后只会运行在一个进程中,其进程名为AndroidManifest.xml文件中指定的应用包名,所有的基本组件都会在这个进程中运行。但是如果需要将某些组件(如Service、Activity等)运行在单独的进程中,就需要用到Android:process属性了。我们可以为android的基础组件指定process属性来指定它们运行在指定进程中。

有什么好处

一般来说,Android应用多进程有三个好处。
1)我们知道Android系统对每个应用进程的内存占用是有限制的,而且占用内存越大的进程,通常被系统杀死的可能性越大。让一个组件运行在单独的进程中,可以减少主进程所占用的内存,降低被系统杀死的概率.
2)如果子进程因为某种原因崩溃了,不会直接导致主程序的崩溃,可以降低我们程序的崩溃率。
3)即使主进程退出了,我们的子进程仍然可以继续工作,假设子进程是推送服务,在主进程退出的情况下,仍然能够保证用户可以收到推送消息。

怎么来实现

对process属性的设置有两种形式:
第一种形式如 android:process=":remote",以冒号开头,冒号后面的字符串原则上是可以随意指定的。如果我们的包名为“com.example.processtest”,则实际的进程名为“com.example.processtest:remote”。这种设置形式表示该进程为当前应用的私有进程,其他应用的组件不可以和它跑在同一个进程中。
第二种情况如 android:process="com.example.processtest.remote",以小写字母开头,表示运行在一个以这个名字命名的全局进程中,其他应用通过设置相同的ShareUID可以和它跑在同一个进程。

有哪些陷阱

其实开启应用内多进程会有一些陷阱,稍微不注意就会陷入其中。我们首先要明确的一点是进程间的内存空间是不可见的。从而,开启多进程后,我们需要面临这样几个问题:
1)Application的多次重建。
2)静态成员的失效。
3)文件共享问题。
4)断点调试问题。

下面介绍一下AIDL

Android 两个进程之间无法直接通信,只能通过系统底层间接通信
能实现这些功能的有AIDL、Binder、Messenger

AIDL是什么,就是Android Interface Description Language,Android接口定义语言

AIDL Binder Messenger之间的区别

分类 IPC(进程间通信) 多线程 多个应用程序
AIDL 支持IPC 支持多线程 支持多个应用程序
Binder 只有IPC 没有多线程 支持多个应用程序
Messenger 只有IPC 没有多线程 不支持多个应用程序

下面展示一下AIDL的使用方法,模拟远程计算功能

1.首先在main下创建一个AIDL文件夹

554A730B-1843-469A-8C87-3CB2F5702FB3.png

2.然后在aidl文件夹中创建aidl文件

创建完aidl文件后都需要build一下

00C41F23-8376-41F6-B912-30EF0078CA49.png
499DEB71-10E5-4A87-BF15-A2071A0669C1.png

3.创建一个service给其他进程(app)使用

AIDL实质就是使用service服务
注意:不要忘了在manifest定义,并且声明他的进程,如下

<service android:name=".service.IRemoteService"
            android:process=":aidl">
            <intent-filter>
                <action android:name="action.aidl"/>
            </intent-filter>
        </service>

注意:这里添加了一个intent-filter,这是因为在不同应用程序之间使用aidl时,如果没有intent-filter,会找不到service,此intent-filter就是为了能够让双方建立联系。


C014D03A-291E-4405-AA9F-7F9EA80564FF.png

public class IRemoteService extends Service {

    /**
     * 当用户绑定到该服务时会 执行
     * @param intent
     * @return
     */
    @Nullable
    @Override
    public IBinder onBind(Intent intent) {
        return iBinder;
    }

    //实现AIDL文件ITest的接口,并生成iBinder供用户绑定使用
    private IBinder iBinder= new ITest.Stub() {
        @Override
        public int add(int num1, int num2) throws RemoteException {
            Log.d("TAG", "已收到远程发送的数据:num1=" + num1 + ";num2=" + num2);
            return num1+num2;
        }
    };
}

4.新建一个ClientModuel以新建一个进程模拟用户,原来的app模拟服务器

同样的方法新建一个AIDL文件夹,在这之下新建一个文件夹,文件夹名需要和服务端的一样(非常重要),将同一个aidl拷贝到该目录下,build一下

E1C46F82-3FB0-46E3-875C-C8B7306C08C0.png

5.实现远程计算功能

在client的MainActivity中绑定远程服务

public class MainActivity extends AppCompatActivity {

    TextView tvSum;
    EditText et1;
    EditText et2;
    Button btn;

    ITest iTest;

    private ServiceConnection connection=new ServiceConnection() {
        //绑定上服务的时候
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            //拿到远程服务
            iTest=ITest.Stub.asInterface(service);
        }
        //断开服务的时候
        @Override
        public void onServiceDisconnected(ComponentName name) {
            //回收资源
            iTest=null;
        }
    };

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

        //软件一启动就绑定服务
        bindService();
    }

    private void bindService() {
        Intent intent=new Intent();
        intent.setComponent(new ComponentName("com.johnson.todo","com.johnson.todo.service.IRemoteService"));//第一个参数包名,第二个参数文件名
        bindService(intent,connection, BIND_AUTO_CREATE);
    }

    private void initWidget() {
        tvSum = (TextView) findViewById(R.id.tv_sum);
        et1 = (EditText) findViewById(R.id.et_1);
        et2 = (EditText) findViewById(R.id.et_2);
        btn= (Button) findViewById(R.id.btn);
        btn.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                add();
            }
        });
    }

    public void add(){
        int num1 = Integer.parseInt(et1.getText().toString());
        int num2 = Integer.parseInt(et2.getText().toString());

        try {
            int sum = iTest.add(num1, num2);
            tvSum.setText(sum+"");
        } catch (RemoteException e) {
            e.printStackTrace();
            tvSum.setText("RemoteException:"+e.toString());
        }

    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
        unbindService(connection);
    }
}

6.最终效果

add.gif
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,924评论 25 708
  • Jianwei's blog 首页 分类 关于 归档 标签 巧用Android多进程,微信,微博等主流App都在用...
    justCode_阅读 5,955评论 1 23
  • 在上一篇Window里提及过IPC,本篇将详细总结IPC,知识点如下: IPC基础及概念多进程模式序列化Seria...
    厘米姑娘阅读 8,167评论 17 29
  • 每日一句:光努力是没有用的,学会用平台优势放大自己的努力。 健康状况:送外卖(QQ计步14711步),生姜泡脚(驱...
    I枫林渡阅读 881评论 0 1
  • 2008年之前,我一直待在一个小县城里面。受到周围环境的影响,上学也只是跟完成任务一样。所学的内容也仅限于课本中的...
    c396267624d0阅读 164评论 0 0