这几天,一位做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. SQL INSERT INTO 语句

    SQL Order By SQL update INSERT INTO 语句 INSERT INTO 语句用于向表格中插入新的行. 语法 INSERT INTO 表名称 VALUES (值1, 值2, ...

  2. iOS10 适配、Xcode8配置总结

    随着iOS10的推送更新到来,勤劳的程序员又在加班加点的搬砖了,为此收集了一些iOS10 更新的技能给大伙参考,不断更新喜欢就star 前沿 一.Xcode8 插件你去哪了 以为是和之前一样 Xcod ...

  3. python第一站

    python 第一站,豆瓣-美国末日评论小爬虫 最近学习python,但是光是看书看视频学习,总是觉得掌握的不够扎实.所以就决定自己去写写爬虫,当带着目的性去学,也许更容易发现自己需要什么.这是酝酿多 ...

  4. oracle 常用函数【转】

    常用Oracle函数 SQL中的单记录函数 1.ASCII 返回与指定的字符对应的十进制数; SQL> select ascii('A') A,ascii('a') a,ascii('0') z ...

  5. 安装springboot时遇到 LoggerFactory is not a Logback LoggerContext but Logback is on the classpath.问题

    将工程外部jar包删除slf4j就可以运行.

  6. asp.net 自带的缓存

    本文导读:在.NET运用中经常用到缓存(Cache)对象.有HttpContext.Current.Cache以及HttpRuntime.Cache,HttpRuntime.Cache是应用程序级别的 ...

  7. CentOS6.6安装vmware workstation报错

    本人系统用的是centos6.6,安装了vmware workstation,启动后一直如下图报错,相关内核已经安装了的,哪位前辈如果解决过这样的问题,麻烦指点指点,小弟在此先谢过了.

  8. openSUSE4.1 安装docker

    安装完Leap后,通过YaST-Software Management安装docker. 安装 docker . yast2-docker. ruby2.1-rubygem-docker-api  . ...

  9. Java内部DNS查询实现和参数设置

    一.Java内部DNS查询 Java使用域名查询时,用的自己内部的域名实现机制,最后都是交给InetAddress去做DNS解析. 源码分析参考:http://blog.arganzheng.me/p ...

  10. LeetCode: Ransom Note

    public class Solution { public boolean canConstruct(String ransomNote, String magazine) { int[] rans ...