小议Android多进程以致Application多次初始化

最近遇到一个bug,当应用加了多进程后,比如总共进程数为N,会出现在startService()onStartCommand()方法会被重复调用(N-1)次的奇怪现象。


祸起

最近遇到两个模块互不相干却受到影响的奇怪问题,一个push模块和一个DaemonProcess模块在一起后,会出现如下现像的问题


当DaemonProcess为应用加了多进程后,比如总共进程数为N,会出现push模块在startService()onStartCommand()方法会被重复调用(N-1)次的奇怪现象。


寻踪

  • 因为我们用的是Jpush的原因,一开始以为是Jpush,但最后发现是因为引用多进程的原因
  • 再寻找下去发现 调用一次startService()onStartCommand()运行多次
  • 而这两者有何关系呢

举证

Demo测试:
首先在Application中申明四个service,其中ServiceAServiceC都各自另开一个进程,ServiceBServiceD都在主进程中,AndroidManifest.xml如下:


        <service android:name=".ServiceA"
                    android:process="com.hujiang.test.servicea"
                 android:exported="true"/>

        <service android:name=".ServiceB"
                 android:exported="false"/>

        <service android:name=".ServiceC"
                 android:process="com.hujiang.test.servicec"
            android:exported="true"/>

        <service android:name=".ServiceD"
                 android:exported="false"/>

此时在Application中启动四个Service


        startService(new Intent(this, ServiceA.class));
        startService(new Intent(this, ServiceB.class));
        startService(new Intent(this, ServiceC.class));
        startService(new Intent(this, ServiceD.class));

同时各Service打下如下log:

 public static final String TAG = ServiceB.class.getSimpleName();
    @Override
    public void onCreate() {
        super.onCreate();
        Log.i(TAG,"onCreate" + "pid:" + android.os.Process.myPid());
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        Log.i(TAG,"onStartCommand" + "pid:" + android.os.Process.myPid());
        return super.onStartCommand(intent, flags, startId);
    }

在log中会发现
onCreate()方法各执行一遍,这个是正常的,但onStartCommand()方法目前执行了三遍,因为共3个进程。

真相

  1. N个进程,N个独立的虚拟机,Application被N次初使化
  2. 处理时应该在Application中分进程初始化数据

剑谱

如下解决方案

 mProcessName = getCurrentProcessName(this);
    Log.i(TAG, "onCreate" + "getProcessName:" + mProcessName);
    Log.i(TAG, "init_all_process");
    if(TextUtils.equals(mProcessName, getPackageName())){
        Log.i(TAG, "init_main_process");
     } else if(TextUtils.equals(getProcessName(this, android.os.Process.myPid()), "com.hujiang.test.servicea")){
        Log.i(TAG, "init_a_process");
     }else if(TextUtils.equals(getProcessName(this, android.os.Process.myPid()), "com.hujiang.test.servicec")){
        Log.i(TAG, "init_c_process");
     }  

获取当前进程名称:

    private String getCurrentProcessName(Context context) {
    int pid = android.os.Process.myPid();
    ActivityManager mActivityManager = (ActivityManager) context
            .getSystemService(Context.ACTIVITY_SERVICE);
    for (ActivityManager.RunningAppProcessInfo appProcess : mActivityManager
            .getRunningAppProcesses()) {
        if (appProcess.pid == pid) {
            return appProcess.processName;
        }
    }
    return null;
}

分别在自己的进程中初始化

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值