android 通过AlarmManager实现守护进程

场景:在app崩溃或手动退出或静默安装后能够自动重启应用activity

前提:得到系统签名 platform.pk8、platform.x509.pem及signapk.jar 三个文件缺一不可(系统源码里有)

步骤一、设置app开机启动(注意新安装的apk必须启动一次才能够接收到开机广播,否则是一直不会启动的)

<!-- 开机启动 -->
    <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
<receiver android:name="com.bihu.tcarspeakdevice.receiver.AutoStartReceiver"
            android:enabled="true">
            <intent-filter>  
        <action android:name="android.intent.action.BOOT_COMPLETED"/>  
        <category android:name="android.intent.category.HOME"/>  
    </intent-filter>  
    </receiver>

 

步骤二、设置闹钟广播(该广播在没有系统签名的情况下很容易被杀死,我的是在apk进行系统签名并且标注system权限下进行OK)

<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="com.xxxxx"
    android:versionCode="21"
    android:versionName="xxxxxa"
    android:sharedUserId="android.uid.system"

 

    private void registerTickAlarm()
    {
        Intent intent = new Intent();
        intent.setAction("com.bihu.tcarspeakdevice.alarmclock");
        PendingIntent pendingIntent = PendingIntent.getBroadcast(this, 0,intent, 0);
        AlarmManager am = (AlarmManager) getSystemService(ALARM_SERVICE);
        am.setRepeating(AlarmManager.RTC_WAKEUP,System.currentTimeMillis(), 5*1000, pendingIntent);
    }

步骤三,注册接收闹钟发的广播

 <receiver 
     android:name="com.xxx.receiver.TickReceiver"
     android:process="com.xxx.TickReceiverProcess">
     <intent-filter>
            <action android:name="com.xxx.xxx.alarmclock" />
        </intent-filter>
 </receiver>
public class TickReceiver extends BroadcastReceiver{

    private String TAG = TickReceiver.class.getSimpleName();
    @Override
    public void onReceive(Context ctx, Intent intent) {
        if(intent.getAction().equals("com.xxx.xxx.alarmclock"))
        {
            //判断activity是否启动并在前台展示
            if(!isForeground(ctx,"com.xxx.xxx.MainActivity"))
            {
                Toast.makeText(ctx, "接收到alarmclock的广播,Activity没运行,立即启动", Toast.LENGTH_SHORT).show();
                Intent intentActivity = new Intent(ctx,MainActivity.class);
                ctx.startActivity(intentActivity);
            }
            else
            {
                Toast.makeText(ctx, "接收到alarmclock的广播,Activity正在运行", Toast.LENGTH_SHORT).show();
            }
        }
    }
     /**
     * 判断某个界面是否在前台
     * 
     * @param context
     * @param className
     *            某个界面名称
     */
    private boolean isForeground(Context context, String className) {
        if (context == null || className==null) {
            return false;
        }

        ActivityManager am = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE);
        List<RunningTaskInfo> list = am.getRunningTasks(1);
        if (list != null && list.size() > 0) {
            ComponentName cpn = list.get(0).topActivity;
            if (className.equals(cpn.getClassName())) {
                return true;
            }
        }
        return false;
    }
}

步骤四、生成unsigned的apk

步骤五、生成带系统签名的apk

将未签名的apk拷贝到签名文件的同一目录下面,通过cmd执行下面的命令

java -jar signapk.jar platform.x509.pem platform.pk8    examble_unsigned.apk    examble_signed.apk

运行即可!

 

******************************************守护进程的其他可能实现方式***************************************************

1、通过双service,一个主apk是主程序A,一个apk中只有service和广播B,A中启动单独进程的B,然后A和B通过互相发送广播进行互相确认,各自的广播中确认自己的主程序是否存活,若不存活,则直接启动activity。

2、NDK方式:父进程中创建子进程,当父进程被kill之后,子进程会被领养(不确定),然后子进程会检测到父进程已死,则启动父进程

3、通过将apk设置为系统桌面

 

大家可以参考微信、QQ的后台存活方式,在查看已运行的程序时他们都是有2个进程2个服务在运行的。

posted @ 2016-02-19 16:30  飞剑  阅读(2175)  评论(0编辑  收藏  举报