进程保活-双进程守护的正确姿势

做车机的开发免不了在后台开很多服务,做车机的APP跟做普通的手机APP区别在于做车机要用到很多的进程间的通信交互等,一个应用死了,可能导致整个机器都运转不正常了,不会进程间的通信怎么可以,撸起来!

双进程守护,肯定得用到多进程,一个应用可以有多个进程的,如何实现多进程呢,service或者activity等都有一个属性android:process,只要设置起名称就可指定该组件运行在该进程中。在这里我们就指定了RemoteService运行在包名.remote这个进程中。

   <service
            android:name=".LocalService"
            android:enabled="true"
            android:exported="true" />
        <service
            android:name=".RemoteService"
            android:enabled="true"
            android:exported="true"
            android:process=".remote"></service>

首先我们在MainActivity中启动两个Activity,

startService(new Intent(this, LocalService.class));
startService(new Intent(this, RemoteService.class));

为什么加双进程守护呢,A死掉B拉A起来,B死掉A拉B起来,那么A和B怎么知道对方有没有死呢?当然用AIDL,就是进程间的通信实现的一种方式。

首先在main下面写一个aidl文件
这里写图片描述

然后在build 下reBuild一下工程,就可生成aidl对应的java文件
这里写图片描述

下面开始正式的代码编写
两个Service的代码基本相同
首先在LocalService中绑定RemoteService

 class MyBinder extends MyAidlInterface.Stub {

        @Override
        public String getServiceName() throws RemoteException {
            return LocalService.class.getSimpleName();
        }
    }

当绑定关系断掉,也就是RemoteService死掉时,在LocalService中把它启动起来并且重新建立绑定关系:

class MyConn implements ServiceConnection {

        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            Log.e("LocalService", "onServiceConnected: " );
        }

        @Override
        public void onServiceDisconnected(ComponentName name) {
            Log.e("LocalService", "onServiceDisconnected: " );
            LocalService.this.startService(new Intent(LocalService.this, RemoteService.class));
            LocalService.this.bindService(new Intent(LocalService.this, RemoteService.class), conn, Context.BIND_IMPORTANT);
        }
    }

核心代码就是上面两段,在RemoteService中的操作基本与之相同。
整体代码如下:

LocalService的代码如下:
package keepalive.com.xn.keepalive;

import android.app.Service;
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.util.Log;

public class LocalService extends Service {
    MyBinder binder;
    MyConn conn;

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

    @Override
    public void onCreate() {
        super.onCreate();
        binder = new MyBinder();
        if (conn == null) {
        conn = new MyConn();
        }

    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        this.bindService(new Intent(this, RemoteService.class), conn, Context.BIND_IMPORTANT);
        return super.onStartCommand(intent, flags, startId);
    }

    class MyBinder extends MyAidlInterface.Stub {

        @Override
        public String getServiceName() throws RemoteException {
            return LocalService.class.getSimpleName();
        }
    }

    /**
     * 连接时需要用到此类
     */
    class MyConn implements ServiceConnection {

        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            Log.e("LocalService", "onServiceConnected: " );
        }

        @Override
        public void onServiceDisconnected(ComponentName name) {
            Log.e("LocalService", "onServiceDisconnected: " );
            LocalService.this.startService(new Intent(LocalService.this, RemoteService.class));
            LocalService.this.bindService(new Intent(LocalService.this, RemoteService.class), conn, Context.BIND_IMPORTANT);
        }
    }
}
RemoteService的代码如下:
package keepalive.com.xn.keepalive;

import android.app.Service;
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.util.Log;

public class RemoteService extends Service {
    MyBinder binder;
    MyConn conn;

    public RemoteService() {
    }

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

    @Override
    public void onCreate() {
        super.onCreate();
        binder = new MyBinder();
        if (conn == null) {
            conn = new MyConn();
        }
    }

    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        this.bindService(new Intent(this, LocalService.class), conn, Context.BIND_IMPORTANT);
        return super.onStartCommand(intent, flags, startId);
    }

    class MyBinder extends MyAidlInterface.Stub {

        @Override
        public String getServiceName() throws RemoteException {
            return RemoteService.class.getSimpleName();
        }
    }

    /**
     * 连接时需要用到此类
     */
    class MyConn implements ServiceConnection {

        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            Log.e(RemoteService.this.toString(), "onServiceConnected: ");
        }

        @Override
        public void onServiceDisconnected(ComponentName name) {
            Log.e(RemoteService.this.toString(), "onServiceDisconnected: ");
            RemoteService.this.startService(new Intent(RemoteService.this, LocalService.class));
            RemoteService.this.bindService(new Intent(RemoteService.this, LocalService.class), conn, Context.BIND_IMPORTANT);
        }
    }
}

思路和代码都是非常简单的,当然还有其他的保活方式,以后再说咯。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Android 进程保活是指应用程序保持在后台运行,即使用户关闭了应用程序的可见界面,仍然能够接收到系统消息或执行定时任务。这通常是通过以下几个机制实现的: 1. **Service**:Service 是 Android 中的一种组件,可以在后台长期运行并执行一些任务,如接收广播、位置更新等。服务可以通过startService()启动,并使用stopSelf()或bindService()来控制其生命周期。 2. **BroadcastReceiver**:应用程序可以通过创建 BroadcastReceiver 来监听特定的系统或自定义事件,当这些事件发生时,即使应用被关闭,广播也会将消息传递给服务或活动,从而间接实现进程保活。 3. **JobScheduler**:Android 6.0(API level 23)以后引入了 JobScheduler API,开发者可以安排定期的任务在后台执行,即使设备处于空闲状态也可以执行。 4. **AlarmManager**:虽然 AlarmManager 不直接支持保活,但结合Service或WakefulBroadcastReceiver可以设置周期性的闹钟,使得应用在指定时间被唤醒,间接实现进程保活。 5. **Foreground Services**:从 Android Oreo (API level 26)开始,Google引入了 Foreground Services,这些服务需要显示通知,即使在后台也能保持运行。 6. **后台任务栈**:通过管理ActivityStack,应用程序可以创建一个堆栈,当用户关闭应用时,任务栈中的顶部活动会被保存,直到用户再次启动应用。 要避免过度保活导致的资源消耗和用户体验下降,还需要注意遵守Android的后台操作策略和权限管理规范。同时,根据用户行为和需求选择合适的保活策略是非常重要的。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值