借鉴了2篇文章:1像素的Activity让应用在息屏后保活 , Android保证service不被杀掉-增强版: 进程保活(根据用户需求慎用)
关于周期网上有好多文章都是提到了“不死的服务”。很多文章提到了做出一个不死的服务。具体提到的方式有:
onStartCommand方法,返回START_STICKY
也就是在service的onstartcommand函数里返回这个值
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
flags = START_STICKY;
return super.onStartCommand(intent, flags, startId);
}
这个时候如果清理的是一个设置了START_STICKY的app,用任务管理器(方法-)还是在设置里面应用管理杀死服务(方法二)都会被杀掉,后来发现在清理软件里吧这个应用加入白名单之后。在使用方法一的时候。acrivity被清理掉了,但是服务不会被清理。,使用方法二的时候,服务会被关闭,然后一段时间又会重新自动出现。即START_STICKY产生的效果。
让你的应用在任务管理器中不显示出来
在配置mainActivity的时候加上这句就可以实现
android:excludeFromRecents="true"
指定应用程序的一些组件运行在不同的进程中
android:process
该进程属性可用于activities、services、content providers和broadcast receivers 和指定的进程中应该执行的特定组件。
在这个例子中,我指定MusicService必须执行在一个单独的“music”的进程:
<manifest ...>
<application
android:icon="@drawable/ic_launcher"
android:label="@string/app_name"
android:theme="@style/Theme.Main" >
<activity
android:name=".MusicActivity"
/>
<service
android:name=".MusicService"
android:process=":music"
/>
</application>
</manifest>
它有什么意义呢?
在这个简短的介绍中,我提到了每一个Android应用程序在运行的时候都有一个不能超出的内存预算值。更精确的说,这限制了它只能在单个基础的进程上运行。换句话说,应用程序的每一个进程都将会有一个专门的内存预算(更不用说其中止时也有更酷的不同的规则)
一个像素保活
原理:监控手机锁屏解锁事件,在屏幕锁屏时启动1个像素的 Activity,在用户解锁时将 Activity 销毁掉。注意该 Activity 需设计成用户无感知。通过该方案,可以使进程的优先级在屏幕锁屏时间由4提升为最高优先级1。
适用范围
适用场景:
适用版本:
具体实现
首先在你的不死Service中注册屏幕亮灭的广播
代码:
package com.luuuzi.notkillservice;
import android.app.Notification;
import android.app.NotificationManager;
import android.app.Service;
import android.content.Intent;
import android.content.IntentFilter;
import android.graphics.BitmapFactory;
import android.os.Build;
import android.os.IBinder;
import android.support.annotation.Nullable;
import android.support.annotation.RequiresApi;
import android.util.Log;
/**
* 需要一直运行的服务,在后台服务去开启广播
* Created by Administrator on 2017/11/13.
*/
public class BackService extends Service {
private String tag = "tag";
private MyBroadCastReceiver receiver;
//NotificationManager notificationManager;
@Nullable
@Override
public IBinder onBind(Intent intent) {
return null;
}
@RequiresApi(api = Build.VERSION_CODES.JELLY_BEAN)
@Override
public void onCreate() {
Log.i(tag,"BackService:启动广播");
//动态注册广播接收者
receiver=new MyBroadCastReceiver();
IntentFilter filter=new IntentFilter();
//添加屏幕熄灭时的action
filter.addAction(Intent.ACTION_SCREEN_OFF);
//添加屏幕亮起时的action
filter.addAction(Intent.ACTION_SCREEN_ON);
//注册广播,并广播内容
registerReceiver(receiver,filter);
super.onCreate();
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
Log.i(tag, "onstart方法");
//销毁广播
//unregisterReceiver(receiver);
return Service.START_STICKY;
}
@Override
public void onDestroy() {
Log.i(tag,"BackService:onDestroy销毁广播");
unregisterReceiver(receiver);
super.onDestroy();
}
}
然后在监听屏幕亮,灭的广播中写入以下代码
package com.luuuzi.notkillservice;
import android.app.ActivityManager;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.util.Log;
/**
* 接收系统发来的广播
* 监听屏幕熄灭和亮起的广播
* Created by Administrator on 2017/11/13.
*/
public class MyBroadCastReceiver extends BroadcastReceiver {
private String tag = "MyBroadCastReceiver";
@Override
public void onReceive(Context context, Intent intent) {
String action = intent.getAction();
if (Intent.ACTION_SCREEN_OFF.equals(action)) {
Intent intent1 = new Intent(context, OnePxActivity.class);
/**
* //activity继承了context重载了startActivity方法,如果使用acitvity中的startActivity,不会有任何限制。
// Calling startActivity() from outside of an Activity context requires the FLAG_ACTIVITY_NEW_TASK flag. Is this really what you want
而如果直接使用context的startActivity则会报上面的错误,
// 根据错误提示信息, 可以得知, 如果要使用这种方式需要打开新的TASK。
*
*/
intent1.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(intent1);
Log.i(tag, "MyBroadCastReceiver:屏幕熄灭,");
} else if (Intent.ACTION_SCREEN_ON.equals(action)) {
//屏幕亮起时,需要关闭一个像素的Activity,
// 同样是通过广播去传递然后关闭
Intent intent2 = new Intent("FinishActivity");
//发送无序广播
context.sendBroadcast(intent2);//发送对应的广播
Log.i(tag, "MyBroadCastReceiver:屏幕亮起");
}
}
}
之后就是一个像素的Activity中编写,在这个Activity中同样需要注册一个广播用于监听接收,屏幕亮起后的广播,然后finish(),一个像素的Activity
package com.luuuzi.notkillservice;
import android.app.Activity;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.content.IntentFilter;
import android.os.Bundle;
import android.support.annotation.Nullable;
import android.util.Log;
import android.view.Gravity;
import android.view.Window;
import android.view.WindowManager;
/**
* 1像素的Activity
* Created by Administrator on 2017/11/13.
*/
public class OnePxActivity extends Activity{
private String tag="OnePxActivity";
//创建一个广播接收者,用于接收广播来关闭Activity
private BroadcastReceiver receiver=new BroadcastReceiver() {
@Override
public void onReceive(Context context, Intent intent) {
Log.i(tag,"监听屏幕亮起后关闭一个像素的Activity");
//收到广播后关闭当前的Activity
OnePxActivity.this.finish();
}
};
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
//获取当前Activity页面的窗体
Window window=getWindow();
//设置属性参数
window.setGravity(Gravity.LEFT |Gravity.TOP);
WindowManager.LayoutParams params=window.getAttributes();
params.x=0;
params.y=0;
params.width=1;
params.height=1;
//给窗体添加属性
window.setAttributes(params);
//动态注册广播,通过该广播来关闭当前Activity(即一个像素的Activity)
registerReceiver(receiver,new IntentFilter("FinishActivity"));
Log.i(tag,"==========OnePxActivity:onCreate");
}
@Override
protected void onDestroy() {
//销毁广播
unregisterReceiver(receiver);
Log.i(tag,"=========OnePxActivity:onDestroy");
super.onDestroy();
}
}
<activity android:name=".OnePxActivity"
android:configChanges="keyboardHidden|orientation|screenSize|navigation|keyboard"
android:excludeFromRecents="true"
android:exported="false"
android:finishOnTaskLaunch="false"
android:launchMode="singleInstance"
android:process=":process"
android:theme="@style/undeadActivityStyle" />
主题配置
<!--1像素的Activity配置的主题-->
<style name="undeadActivityStyle">
<!-- 背景设置为透明 -->
<item name="android:windowBackground">@android:color/transparent</item>
<!-- 是否有边框 -->
<item name="android:windowFrame">@null</item>
<item name="android:windowNoTitle">true</item>
<!-- 是否浮动在界面上 -->
<item name="android:windowIsFloating">true</item>
<!-- 是否透明 :透明-->
<item name="android:windowIsTranslucent">true</item>
<!-- 窗体上面是否有遮盖 无遮盖 -->
<item name="android:windowContentOverlay">@null</item>
<!-- 为窗体的Enter和Exit设置动画 -->
<item name="android:windowAnimationStyle">@null</item>
<!-- 是否禁止窗体显示前显示的View -->
<item name="android:windowDisablePreview">true</item>
<item name="android:windowNoDisplay">false</item>
</style>
最后去开启你的一直运行的后台服务
//开启服务
Log.i(tag,"开启服务");
Intent intent = new Intent(MainActivity.this, BackService.class);
startService(intent);
以上就实现了屏幕锁屏后应用不会被杀死掉的逻辑
方法2
让通过notification挂起在通知栏上,实现前台的服务
1.创建长期运行Service
/**
* 需要一直运行的服务,在后台服务去开启广播
* 创建一个通知,让其长期在后台运行
* Created by Administrator on 2017/11/13.
*/
public class BackService extends Service {
private ScreenReceiver receiver;
@Nullable
@Override
public IBinder onBind(Intent intent) {
return null;
}
@RequiresApi(api = Build.VERSION_CODES.JELLY_BEAN)
@Override
public void onCreate() {
//将通知栏挂起
Intent intent = new Intent();
startForeground(10, createNotification(getApplicationContext()));
super.onCreate();
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
return Service.START_STICKY;
}
@Override
public void onDestroy() {
super.onDestroy();
}
/**
* 创建通知
*
* @return
*/
@RequiresApi(api = Build.VERSION_CODES.JELLY_BEAN)
public Notification createNotification(Context context) {
// 1.通过上下文对象吧获取通知的管理者
NotificationManager nm = (NotificationManager) getSystemService(NOTIFICATION_SERVICE);
//2.定义通知,链式调用(返回值一直都是自己就可以链式调用) 高版本的写法
Notification notification = new Notification.Builder(context)
.setContentTitle("好实再联盟") //标题
.setContentText("语音播报已开启") //内容
.setSmallIcon(R.mipmap.logo) //图标
.setLargeIcon(BitmapFactory.decodeResource(getResources(), R.mipmap.logo)).build();
//notification.flags = Notification.FLAG_NO_CLEAR|Notification.FLAG_ONGOING_EVENT;
//设置标记
notification.flags = Notification.FLAG_NO_CLEAR | Notification.FLAG_ONGOING_EVENT;
return notification;
}
}
2,创建一个接收系统广播的广播接收者
/**
* 接收系统发来的广播,然后去判断后台进程是否被杀死,
* 如果被杀死了则重启Service
* Created by Administrator on 2017/11/13.
*/
public class MyBroadCastReceiver extends BroadcastReceiver {
private String tag = "MyBroadCastReceiver";
@Override
public void onReceive(Context context, Intent intent) {
//记录是否被杀死,如果被杀死了,咋重新启动
boolean isServiceRunning = false;
ActivityManager manager = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE);
for (ActivityManager.RunningServiceInfo service : manager.getRunningServices(Integer.MAX_VALUE)) {
if ("com.hsz88.merchant.service.BackService".equals(service.service.getClassName())) {
Log.i(tag, "服务没被杀死");
isServiceRunning = true;
}
}
if (!isServiceRunning) {
Log.i(tag, "服务被杀死了");
Intent i = new Intent(context, BackService.class);
i.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startService(i);
}
}
}
3.启动Service和BroadCastReceiver
/**
* 启动Service
*/
//开启服务
Log.i(tag, "开启服务");
Intent intent = new Intent(mContext, BackService.class);
startService(intent);
//启动监听系统时间广播
receiver = new MyBroadCastReceiver();
IntentFilter filter = new IntentFilter();
//监听系统广播,该广播一分钟发一次
filter.addAction(Intent.ACTION_TIME_TICK);
//启动该广播
registerReceiver(receiver, filter);