这几天,一位做Android的朋友和我探讨了一个问题:因为业务需求的原因,在自己的App长时间不使用被kill掉之后,如何让它再重新运行起来。

虽然,我本身很排斥这种做法,有点类似“流氓软件”的行为,但是还是查询了资料,大概想了一个实现的方式,和大家一起分享。

其实,这个问题可以简单的看作:如何编写一个守护进程

使用C/C++编写一个守护进程的.so程序,Android端通过JNI调用。该进程监听当前的目标程序进程,如果目标程序被kill掉了,再重新start一下,大概的思路就是这样。伪代码如下:

int pid = fork();
if(pid <)
{
printf("Error\n");
exit();
}
else if(pid > )
{
printf("this is father\n");
exit();
}
else if(pid == ) // child1
{
setsid();
int pid = fork();
if(pid == ) // child2
{
while()
{
if(App is killed)
start App again;
}
}
}

这其中,缺少了:

①unmask(),因为从父进程继承下来了一些东西,所以需要设置一定的权限进行操作。

②chdir("/")。

③关闭从父进程继承而来的不需要的文件描述符,否则就可能造成资源的浪费。

为什么要进行两次fork()?

在这段伪代码中,进行了两次fork()操作,其实很多文章在描述守护进程时都只是仅仅的进行了一次fork操作就结束了,这里我想仔细讲一讲原因。

首先,setsid()函数的作用。一般而言,父进程和子进程都处在一个session当中,父进程是session的领头进程。如果当父进程(领头进程)被杀死之后,那么同一个session中的所有进程都会被杀死,或者成为孤儿进程而被init托管。所以,我们需要让子进程调用setsid(),创建一个新的session并将自己设置为该session的领头进程(若领头进程调用setsid()则没有任何效果)。这样,如果父进程被kill掉,因为他们并不在一个session中,所以子进程仍然可以继续执行。由于session对控制终端的独占性,进程同时与控制终端脱离。

session中包含了很多东西,如:控制终端、进程组等等。如果我们只fork()一次,将第一个由父进程创建出的子进程分离出来作为守护进程,一般情况下也是没有什么问题的。但是,如果此时通过什么方式通过此进程创建出了一个与自己的session相关联的控制终端,那么则会产生一定的影响。所以,则有了第二次fork()的意义。第二次fork使用子进程创建出了一个“孙子进程”,并且我们还是使用第一个父进程创建出的子进程进行setsid()但它并不是守护进程,该孙子进程进行守护进程相关的操作。这就有效的避免了上文提到的问题的产生,因为子进程创建出了一个新的session,并且作为该session的领头进程,同时这个session包含了该孙子进程且它并不是领头进程。那么该孙子进程就永远无法创建出一个控制终端,也就没有任何影响。

存在的问题

代码中使用轮询的方式来查询,App进程是否被kill了,这样效率十分低下,会导致手机的电量损耗很快。是否可以通过进程间通信的方式,如:socket等等,进行相关操作。

Android守护进程的更多相关文章

  1. 创建Android守护进程(底层服务)【转】

    本文转载自:https://blog.csdn.net/myfriend0/article/details/80016739 创建Android守护进程(底层服务) 前言 Android底层服务,即运 ...

  2. Android 通过JNI实现守护进程,使得Service服务不被杀死

    来自: http://finalshares.com/read-7306 转载请注明出处: http://blog.csdn.net/yyh352091626/article/details/5054 ...

  3. android 通过AlarmManager实现守护进程

    场景:在app崩溃或手动退出或静默安装后能够自动重启应用activity 前提:得到系统签名 platform.pk8.platform.x509.pem及signapk.jar 三个文件缺一不可(系 ...

  4. android双进程守护,让程序崩溃后一定可以重启

    由于我们做的是机器人上的软件,而机器人是24小时不间断服务的,这就要求我们的软件不能退出到系统桌面.当然最好是能够做到程序能够不卡顿,不崩溃,自己不退出.由于我们引用了很多第三方的开发包,也不能保证他 ...

  5. Android init进程概述

    init进程,其程序位于根文件系统中,在kernle自行启动后,其中的 start_kernel 函数把根文件系统挂载到/目录后,在 rest_init 函数中通过 kernel_thread(ker ...

  6. APP中一种在Java层实现的简单守护进程方式

    转载请把头部出处链接和尾部二维码一起转载,本文出自逆流的鱼yuiop:http://blog.csdn.net/hejjunlin/article/details/52779986 守护进程是一个黑色 ...

  7. ANDROID init进程

    init简要 init是Android上启动的第一个用户态进程. 执行序列是: start_kernel() -> rest_init() -> kernel_init() -> i ...

  8. 一篇文章了解相见恨晚的 Android Binder 进程间通讯机制【转】

    本文转载自:https://blog.csdn.net/freekiteyu/article/details/70082302 Android-Binder进程间通讯机制 概述 最近在学习Binder ...

  9. dhcpcd守护进程分析【转】

    本文转载自;http://blog.csdn.net/lishanmin11/article/details/37930073 最近在调android ethernet功能,android本身不带 e ...

随机推荐

  1. iOS相关笔记

    #协议[1] [2] @property (nonatomic, assign) id<EveryFrameDelegate> delegate; 表明,这个delegate是一个需要实现 ...

  2. Nodejs:Path对象

    模块Path: path.normalize(url) path.join(url1, url2, ..) path.resolve(url1, url2, ..) path.relative(url ...

  3. Android--数据存储

    1. 文件存储指定的文件名不可以包含路径, 默认保存到 /data/data/<package name>/files/ 目录下 2. SharedPreferences存储使用键值对的方 ...

  4. iOS版本更新的App提交审核发布流程

    http://www.2cto.com/kf/201502/378698.html 版本更新的App和新App的发布提交流程略有不同,新的App需要在开发者账号里准备发布证书,添加App的id,关联描 ...

  5. Git和Github简单教程(转)

    这篇文章只挑一部分命令来讲,差不多够用就行的程度.如果需要其他命令,到时候再去其他地方了解就行了先在Github上写好再搬过来的:本文Github链接 目录: 零.Git是什么 一.Git的主要功能: ...

  6. 我理解的IOC技术在Java和C#中比较分析

    一直想用心写这个系列的文章,其实看得越多,也就越觉得自己在这方面的功力太浅,也就越不想班门弄斧啦,作为一个开篇,我想把这个技术深层次化,在之前的.net的一个MVC系列文章其实已经涉及到了,只是.ne ...

  7. linux命令日常总结

    1.date 显示系统日期2. mkdir xx 创建xx目录 rmdir xx 删除xx目录(空目录) rm -rf xx 删除xx目录(非空目录) 3. vi xx 创建某文件 写入-->e ...

  8. Redmine新建问题速度慢

    Redmine有时候新建问题 ,更新指派人的时候反应很慢, 很大原因应该是发送邮件方式不对. 1.一种方式是改为异步发送      2.另外检测到Redmine日志 ,会发现发送邮件失败 ,也会导致发 ...

  9. XSLT简介

    什么是? http://www.w3school.com.cn/xsl/xsl_intro.asp XSLT 是一种用于将 XML 文档转换为 XHTML 文档或其他 XML 文档的语言. XPath ...

  10. Android四大核心组件之BroadCastReceiver

    实验内容 实现BroadCast发送和接受 通过BroadCast传递信息 动态注册和注销BroadCast 实验要求 实现BroadCast发送和接受 通过BroadCast传递信息 动态注册和注销 ...