进程优先级
1.前台进程
2.可见进程
3.服务进程
4.后台进程
5.空进程
内存阈值查看(每个手机不一样,阈值单位:4KB)
cat /sys/module/lowmemorykiller/parameters/minfree
进程优先级查看
cat /proc/进程id/oom_adj
ProcessState级别 | 取值 | 解释 |
---|---|---|
PROCESS_STATE_CACHED_EMPTY | 16 | 进程处于cached状态,且为空进程 |
PROCESS_STATE_CACHED_ACTIVITY_CLIENT | 15 | 进程处于cached状态,且为另一个cached进程(内含Activity)的client进程 |
PROCESS_STATE_CACHED_ACTIVITY | 14 | 进程处于cached状态,且内含Activity |
PROCESS_STATE_LAST_ACTIVITY | 13 | 后台进程,且拥有上一次显示的Activity |
PROCESS_STATE_HOME | 12 | 后台进程,且拥有home Activity |
PROCESS_STATE_RECEIVER | 11 | 后台进程,且正在运行receiver |
PROCESS_STATE_SERVICE | 10 | 后台进程,且正在运行service |
PROCESS_STATE_HEAVY_WEIGHT | 9 | 后台进程,但无法执行restore,因此尽量避免kill该进程 |
PROCESS_STATE_BACKUP | 8 | 后台进程,正在运行backup/restore操作 |
PROCESS_STATE_IMPORTANT_BACKGROUND | 7 | 对用户很重要的进程,用户不可感知其存在 |
PROCESS_STATE_IMPORTANT_FOREGROUND | 6 | 对用户很重要的进程,用户可感知其存在 |
PROCESS_STATE_TOP_SLEEPING | 5 | 与PROCESS_STATE_TOP一样,但此时设备正处于休眠状态 |
PROCESS_STATE_FOREGROUND_SERVICE | 4 | 拥有一个前台Service |
PROCESS_STATE_BOUND_FOREGROUND_SERVICE | 3 | 拥有一个前台Service,且由系统绑定 |
PROCESS_STATE_TOP | 2 | 拥有当前用户可见的top Activity |
PROCESS_STATE_PERSISTENT_UI | 1 | persistent系统进程,并正在执行UI操作 |
PROCESS_STATE_PERSISTENT | 0 | persistent系统进程 |
PROCESS_STATE_NONEXISTENT | -1 | 不存在的进程 |
ADJ 调度算法的核心方法
-
updateOomAdjLocked
:更新adj,当目标进程为空,或者被杀则返回false;否则返回true; -
computeOomAdjLocked
:计算adj,返回计算后RawAdj值; -
applyOomAdjLocked
:使用adj,当需要杀掉目标进程则返回false;否则返回true。
保活方式:
1.Activity 1像素保活
原理:手机关闭屏幕时,偷偷创建一个Activity,让应用成为前台进程;打开屏幕时,关闭该Activity。
缺点:存在一个Activity不够干净。同时也需要在锁屏后才能提权。
/**
* 屏幕开启关闭广播
*/
public class KeepReceiver extends BroadcastReceiver {
private static final String TAG = "KeepReceiver";
@Override
public void onReceive(Context context, Intent intent) {
String action = intent.getAction();
Log.e(TAG, "onReceive: " + action);
if (TextUtils.equals(action, Intent.ACTION_SCREEN_OFF)) {
// 关闭屏幕时 开启1像素activity
KeepManager.getInstance().startKeep(context);
} else if (TextUtils.equals(action, Intent.ACTION_SCREEN_ON)) {
// 打开屏幕时 关闭1像素activity
KeepManager.getInstance().finishKeep();
}
}
}
/**
* 1像素Activity
*/
public class KeepActivity extends Activity {
private static final String TAG = "KeepActivity";
@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
Log.e(TAG, "启动keep");
Window window = getWindow();
//放在左上角
window.setGravity(Gravity.START | Gravity.TOP);
WindowManager.LayoutParams params = window.getAttributes();
//设置宽高
params.width = 1;
params.height = 1;
//设置起始坐标
params.x = 0;
params.y = 0;
window.setAttributes(params);
// KeepActivity 创建一个弱引用
KeepManager.getInstance().setKeep(this);
}
@Override
protected void onDestroy() {
super.onDestroy();
Log.e(TAG, "关闭keep");
}
}
/**
* 1像素保活管理
*/
public class KeepManager {
private static final KeepManager mInstance = new KeepManager();
//广播
private KeepReceiver mKeepReceiver;
//弱引用
private WeakReference<Activity> mKeepActivity;
private KeepManager() {
}
public static KeepManager getInstance() {
return mInstance;
}
/**
* 注册 开屏 关屏 广播
*
* @param context
*/
public void registerKeep(Context context) {
IntentFilter filter = new IntentFilter();
filter.addAction(Intent.ACTION_SCREEN_ON);
filter.addAction(Intent.ACTION_SCREEN_OFF);
mKeepReceiver = new KeepReceiver();
context.registerReceiver(mKeepReceiver, filter);
}
/**
* 注销 广播接收者
*
* @param context
*/
public void unregisterKeep(Context context) {
if (mKeepReceiver != null) {
context.unregisterReceiver(mKeepReceiver);
}
}
/**
* 开启1像素Activity
*
* @param context
*/
public void startKeep(Context context) {
Intent intent = new Intent(context, KeepActivity.class);
// 结合 taskAffinity 一起使用 在指定栈中创建这个activity
intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
context.startActivity(intent);
}
/**
* 关闭1像素Activity
*/
public void finishKeep() {
if (mKeepActivity != null) {
Activity activity = mKeepActivity.get();
if (activity != null) {
activity.finish();
}
mKeepActivity = null;
}
}
/**
* 设置弱引用
*
* @param keep
*/
public void setKeep(KeepActivity keep) {
mKeepActivity = new WeakReference<Activity>(keep);
}
}
//使用
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
KeepManager.getInstance().registerKeep(this);
}
@Override
protected void onDestroy() {
KeepManager.getInstance().unregisterKeep(this);
super.onDestroy();
}
}
<!--excludeFromRecents=true 表示在任务管理器看不到-->
<!--taskAffinity 不同的任务栈-->
<activity
android:name=".activity.KeepActivity"
android:excludeFromRecents="true"
android:taskAffinity="com.cq.daemon.keep"
android:theme="@style/KeepTheme" />
<style name="KeepTheme">
<!--背景-->
<item name="android:windowBackground">@null</item>
<!--是否透明-->
<item name="android:windowIsTranslucent">true</item>
</style>
2.前台 Service 保活
原理:启动一个前台服务,从而拉高整个应用的优先级。
缺点:API >=26 后暂时没有方式能够隐藏通知
/**
* 前台服务保活
*/
public class ForegroundService extends Service {
private static final String TAG = "ForegroundService";
private static final int SERVICE_ID = 1;
@Nullable
@Override
public IBinder onBind(Intent intent) {
return null;
}
@Override
public void onCreate() {
super.onCreate();
Log.e(TAG, "ForegroundService 服务创建了");
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.JELLY_BEAN_MR2) {//4.3以下
//将service设置成前台服务,并且不显示通知栏消息
startForeground(SERVICE_ID, new Notification());
} else if (Build.VERSION.SDK_INT < Build.VERSION_CODES.O) { //Android4.3-->Android7.0
//将service设置成前台服务
startForeground(SERVICE_ID, new Notification());
//删除通知栏消息
startService(new Intent(this, InnerService.class));
} else { // 8.0 及以上
//通知栏消息需要设置channel
NotificationManager manager = (NotificationManager) getSystemService(NOTIFICATION_SERVICE);
//NotificationManager.IMPORTANCE_MIN 通知栏消息的重要级别 最低,不让弹出
//IMPORTANCE_MIN 前台时,在阴影区能看到,后台时 阴影区不消失,增加显示 IMPORTANCE_NONE时 一样的提示
//IMPORTANCE_NONE app在前台没有通知显示,后台时有
NotificationChannel channel = new NotificationChannel("channel", "xx", NotificationManager.IMPORTANCE_NONE);
if (manager != null) {
manager.createNotificationChannel(channel);
Notification notification = new NotificationCompat.Builder(this, "channel").build();
//将service设置成前台服务,8.x退到后台会显示通知栏消息,9.0会立刻显示通知栏消息
startForeground(SERVICE_ID, notification);
}
}
}
public static class InnerService extends Service {
@Override
public void onCreate() {
super.onCreate();
Log.e(TAG, "InnerService 服务创建了");
// 让服务变成前台服务
startForeground(SERVICE_ID, new Notification());
// 关闭自己
stopSelf();
}
@Nullable
@Override
public IBinder onBind(Intent intent) {
return null;
}
}
}
// 前台服务保活
startService(new Intent(this, ForegroundService.class));
<!--前台服务权限-->
<uses-permission android:name="android.permission.FOREGROUND_SERVICE"/>
<!--前台service保活-->
<service android:name=".service.ForegroundService" />
<service android:name=".service.ForegroundService$InnerService" />
拉活方式:
1.广播拉活
在发生特定系统事件时,系统会发出广播,通过在 AndroidManifest 中静态注册对应的广播监听器,即可在发生响应事件时拉活。
但是从android 7.0开始,对广播进行了限制,而且在8.0更加严格https://developer.android.google.cn/about/versions/oreo/background.html#broadcasts
可静态注册广播列表:
https://developer.android.google.cn/guide/components/broadcast-exceptions.html
“全家桶”拉活
有多个app在用户设备上安装,只要开启其中一个就可以将其他的app也拉活。比如手机里装了手Q、QQ空间、兴趣部落等等,那么打开任意一个app后,其他的app也都会被唤醒。
2.Service系统机制拉活
START_STICKY:
“粘性”。如果service进程被kill掉,保留service的状态为开始状态,但不保留递送的intent对象。随后系统会尝试重新创建service,由于服务状态为开始状态,所以创建服务后一定会调用onStartCommand(Intent,int,int)方法。如果在此期间没有任何启动命令被传递到service,那么参数Intent将为null。
START_NOT_STICKY:
“非粘性的”。使用这个返回值时,如果在执行完onStartCommand后,服务被异常kill掉,系统不会自动重启该服务。
START_REDELIVER_INTENT:
重传Intent。使用这个返回值时,如果在执行完onStartCommand后,服务被异常kill掉,系统会自动重启该服务,并将Intent的值传入。
START_STICKY_COMPATIBILITY:
START_STICKY的兼容版本,但不保证服务被kill后一定能重启。
只要 targetSdkVersion 不小于5,就默认是 START_STICKY。
但是某些ROM 系统不会拉活。并且经过测试,Service 第一次被异常杀死后很快被重启,第二次会比第一次慢,第三次又会比前一次慢,一旦在短时间内 Service 被杀死4-5次,则系统不再拉起。
/**
* Service系统机制拉活
*/
public class StickyService extends Service {
@Nullable
@Override
public IBinder onBind(Intent intent) {
return null;
}
//START_STICKY
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
//只要 targetSdkVersion 不小于5,就默认是 START_STICKY。
return super.onStartCommand(intent, flags, startId);
}
}
//sticky
startService(new Intent(this, StickyService.class));
3.账户同步拉活
优点:系统唤醒,比较稳定
缺点:时间不能把控
public class AccountHelper {
private static final String TAG = "AccountHelper";
private static final String ACCOUNT_TYPE = "com.cq.daemon.account";
/**
* 添加账号
*
* @param context
*/
public static void addAccount(Context context) {
AccountManager accountManager = (AccountManager) context.getSystemService(Context.ACCOUNT_SERVICE);
// 获得此类型的账户
// 需要增加权限 GET_ACCOUNTS
Account[] accounts = accountManager.getAccountsByType(ACCOUNT_TYPE);
if (accounts.length > 0) {
Log.e(TAG, "账户已存在");
return;
}
Account account = new Account("cq", ACCOUNT_TYPE);
// 给这个账户类型添加一个账户
// 需要增加权限 AUTHENTICATE_ACCOUNTS
accountManager.addAccountExplicitly(account, "cq", new Bundle());
}
/**
* 设置账户自动同步
*/
public static void autoSync() {
Account account = new Account("cq", ACCOUNT_TYPE);
// 下面三个都需要同一个权限 WRITE_SYNC_SETTINGS
// 设置同步
ContentResolver.setIsSyncable(account, "com.cq.daemon.provider", 1);
// 自动同步
ContentResolver.setSyncAutomatically(account, "com.cq.daemon.provider", true);
// 设置同步周期
ContentResolver.addPeriodicSync(account, "com.cq.daemon.provider", new Bundle(), 1);
}
}
/**
* 创建 可添加用户
*/
public class AuthenticationService extends Service {
private AccountAuthenticator accountAuthenticator;
@Nullable
@Override
public IBinder onBind(Intent intent) {
return accountAuthenticator.getIBinder();
}
@Override
public void onCreate() {
super.onCreate();
accountAuthenticator = new AccountAuthenticator(this);
}
public static class AccountAuthenticator extends AbstractAccountAuthenticator {
public AccountAuthenticator(Context context) {
super(context);
}
@Override
public Bundle editProperties(AccountAuthenticatorResponse response, String accountType) {
return null;
}
@Override
public Bundle addAccount(AccountAuthenticatorResponse response, String accountType, String authTokenType,
String[] requiredFeatures, Bundle options) throws NetworkErrorException {
return null;
}
@Override
public Bundle confirmCredentials(AccountAuthenticatorResponse response, Account account,
Bundle options) throws NetworkErrorException {
return null;
}
@Override
public Bundle getAuthToken(AccountAuthenticatorResponse response, Account account,
String authTokenType, Bundle options) throws NetworkErrorException {
return null;
}
@Override
public String getAuthTokenLabel(String authTokenType) {
return null;
}
@Override
public Bundle updateCredentials(AccountAuthenticatorResponse response, Account account,
String authTokenType, Bundle options) throws NetworkErrorException {
return null;
}
@Override
public Bundle hasFeatures(AccountAuthenticatorResponse response, Account account,
String[] features) throws NetworkErrorException {
return null;
}
}
}
public class SyncProvider extends ContentProvider {
@Override
public boolean onCreate() {
return false;
}
@Nullable
@Override
public Cursor query(@NonNull Uri uri, @Nullable String[] projection, @Nullable String selection,
@Nullable String[] selectionArgs, @Nullable String sortOrder) {
return null;
}
@Nullable
@Override
public String getType(@NonNull Uri uri) {
return null;
}
@Nullable
@Override
public Uri insert(@NonNull Uri uri, @Nullable ContentValues values) {
return null;
}
@Override
public int delete(@NonNull Uri uri, @Nullable String selection, @Nullable String[] selectionArgs) {
return 0;
}
@Override
public int update(@NonNull Uri uri, @Nullable ContentValues values, @Nullable String selection,
@Nullable String[] selectionArgs) {
return 0;
}
}
public class SyncService extends Service {
private SyncAdapter mSyncAdapter;
private static final String TAG = "SyncService";
@Nullable
@Override
public IBinder onBind(Intent intent) {
return mSyncAdapter.getSyncAdapterBinder();
}
@Override
public void onCreate() {
super.onCreate();
mSyncAdapter = new SyncAdapter(getApplicationContext(), true);
}
public static class SyncAdapter extends AbstractThreadedSyncAdapter {
public SyncAdapter(Context context, boolean autoInitialize) {
super(context, autoInitialize);
}
@Override
public void onPerformSync(Account account, Bundle extras, String authority, ContentProviderClient provider, SyncResult syncResult) {
Log.e(TAG, "同步账户");
//与互联网 或者 本地数据库同步账户
}
}
}
<!--account_authenticator.xml-->
<?xml version="1.0" encoding="utf-8"?>
<account-authenticator xmlns:android="http://schemas.android.com/apk/res/android"
android:accountType="com.cq.daemon.account"
android:icon="@mipmap/ic_launcher"
android:label="@string/app_name" />
<!--android:label="Daemon" 不能这么写-->
<!--sync_adapter.xml-->
<?xml version="1.0" encoding="utf-8"?>
<sync-adapter xmlns:android="http://schemas.android.com/apk/res/android"
android:accountType="com.cq.daemon.account"
android:allowParallelSyncs="false"
android:contentAuthority="com.cq.daemon.provider"
android:isAlwaysSyncable="true"
android:userVisible="true" />
<!-- allowParallelSyncs 是否支持多账号同时同步-->
<!--contentAuthority 指定要同步的 ContentProvider-->
<!--android:userVisible 显示开关按钮 给用户控制-->
<!--账户同步拉活权限-->
<uses-permission android:name="android.permission.WRITE_SYNC_SETTINGS" />
<uses-permission
android:name="android.permission.AUTHENTICATE_ACCOUNTS"
android:maxSdkVersion="22" />
<uses-permission
android:name="android.permission.GET_ACCOUNTS"
android:maxSdkVersion="22" />
<!--账户同步拉活-->
<service android:name=".account.AuthenticationService">
<!--<action android:name="android.accounts.AccountAuthenticator" /> 让系统能够找到这个账户服务-->
<intent-filter>
<action android:name="android.accounts.AccountAuthenticator" />
</intent-filter>
<meta-data
android:name="android.accounts.AccountAuthenticator"
android:resource="@xml/account_authenticator" />
</service>
<!--同步操作-->
<service android:name=".account.SyncService">
<intent-filter>
<action android:name="android.content.SyncAdapter" />
</intent-filter>
<meta-data
android:name="android.content.SyncAdapter"
android:resource="@xml/sync_adapter" />
</service>
<provider
android:name=".account.SyncProvider"
android:authorities="com.cq.daemon.provider" />
// 账户拉活
AccountHelper.addAccount(this);
AccountHelper.autoSync();
4.JobScheduler 拉活
JobScheduler允许在特定状态与特定时间间隔周期执行任务。可以利用它的这个特点完成保活的功能,效果即开启一个定时器,与普通定时器不同的是其调度由系统完成。
同样在某些ROM可能并不能达到需要的效果
public class MyJobService extends JobService {
private static final String TAG = "MyJobService";
public static void startJob(Context context) {
JobScheduler jobScheduler = (JobScheduler) context.getSystemService(Context.JOB_SCHEDULER_SERVICE);
//setPersisted 在设备重启依然执行
// 需要增加权限 RECEIVE_BOOT_COMPLETED
JobInfo.Builder builder = new JobInfo.Builder(8, new ComponentName(context.getPackageName(),
MyJobService.class.getName())).setPersisted(true);
// 小于7.0
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.N) {
// 每隔 1s 执行一次 job
// 版本23 开始 进行了改进,最小周期为 5s
builder.setPeriodic(1000);
} else {
// 延迟执行任务
builder.setMinimumLatency(1000);
}
jobScheduler.schedule(builder.build());
}
@Override
public boolean onStartJob(JobParameters params) {
Log.e(TAG, "onStartJob");
// 如果7.0以上 轮询
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
startJob(this);
}
return false;
}
@Override
public boolean onStopJob(JobParameters params) {
return false;
}
}
<!--job拉活需要的权限-->
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
<!--job拉活-->
<service
android:name=".jobscheduler.MyJobService"
android:permission="android.permission.BIND_JOB_SERVICE" />
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
MyJobService.startJob(this);
}
5.双进程+JobScheduler守护
双进程没什么太大用处了,7.0以上系统杀的是进程组了
public class LocalService extends Service {
public static final String TAG = "ProcessService";
private ServiceConnection serviceConnection;
private MyBinder myBinder;
private static final int SERVICE_ID = 17;
class MyBinder extends IMyAidlInterface.Stub {
}
@Nullable
@Override
public IBinder onBind(Intent intent) {
return myBinder;
}
@Override
public void onCreate() {
super.onCreate();
myBinder = new MyBinder();
serviceConnection = new MyServiceConnection();
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.JELLY_BEAN_MR2) {//4.3以下
//将service设置成前台服务,并且不显示通知栏消息
startForeground(SERVICE_ID, new Notification());
} else if (Build.VERSION.SDK_INT < Build.VERSION_CODES.O) { //Android4.3-->Android7.0
//将service设置成前台服务
startForeground(SERVICE_ID, new Notification());
//删除通知栏消息
startService(new Intent(this, InnerService.class));
} else { // 8.0 及以上
//通知栏消息需要设置channel
NotificationManager manager = (NotificationManager) getSystemService(NOTIFICATION_SERVICE);
//NotificationManager.IMPORTANCE_MIN 通知栏消息的重要级别 最低,不让弹出
//IMPORTANCE_MIN 前台时,在阴影区能看到,后台时 阴影区不消失,增加显示 IMPORTANCE_NONE时 一样的提示
//IMPORTANCE_NONE app在前台没有通知显示,后台时有
NotificationChannel channel = new NotificationChannel("channel", "xx", NotificationManager.IMPORTANCE_NONE);
if (manager != null) {
manager.createNotificationChannel(channel);
Notification notification = new NotificationCompat.Builder(this, "channel").build();
//将service设置成前台服务,8.x退到后台会显示通知栏消息,9.0会立刻显示通知栏消息
startForeground(SERVICE_ID, notification);
}
}
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
bindService(new Intent(this, RemoteService.class), serviceConnection, BIND_AUTO_CREATE);
return super.onStartCommand(intent, flags, startId);
}
class MyServiceConnection implements ServiceConnection {
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
}
@Override
public void onServiceDisconnected(ComponentName name) {
Log.e(TAG, "localService 可能被杀死了,拉活");
startService(new Intent(LocalService.this, RemoteService.class));
bindService(new Intent(LocalService.this, RemoteService.class),
serviceConnection, BIND_AUTO_CREATE);
}
}
public static class InnerService extends Service {
@Override
public void onCreate() {
super.onCreate();
Log.e(TAG, "InnerService 服务创建了");
// 让服务变成前台服务
startForeground(SERVICE_ID, new Notification());
// 关闭自己
stopSelf();
}
@Nullable
@Override
public IBinder onBind(Intent intent) {
return null;
}
}
}
public class RemoteService extends Service {
public static final String TAG = "ProcessService";
private ServiceConnection serviceConnection;
private MyBinder myBinder;
private static final int SERVICE_ID = 17;
class MyBinder extends IMyAidlInterface.Stub {
}
@Nullable
@Override
public IBinder onBind(Intent intent) {
return myBinder;
}
@Override
public void onCreate() {
super.onCreate();
myBinder = new MyBinder();
serviceConnection = new MyServiceConnection();
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.JELLY_BEAN_MR2) {//4.3以下
//将service设置成前台服务,并且不显示通知栏消息
startForeground(SERVICE_ID, new Notification());
} else if (Build.VERSION.SDK_INT < Build.VERSION_CODES.O) { //Android4.3-->Android7.0
//将service设置成前台服务
startForeground(SERVICE_ID, new Notification());
//删除通知栏消息
startService(new Intent(this, InnerService.class));
} else { // 8.0 及以上
//通知栏消息需要设置channel
NotificationManager manager = (NotificationManager) getSystemService(NOTIFICATION_SERVICE);
//NotificationManager.IMPORTANCE_MIN 通知栏消息的重要级别 最低,不让弹出
//IMPORTANCE_MIN 前台时,在阴影区能看到,后台时 阴影区不消失,增加显示 IMPORTANCE_NONE时 一样的提示
//IMPORTANCE_NONE app在前台没有通知显示,后台时有
NotificationChannel channel = new NotificationChannel("channel", "xx", NotificationManager.IMPORTANCE_NONE);
if (manager != null) {
manager.createNotificationChannel(channel);
Notification notification = new NotificationCompat.Builder(this, "channel").build();
//将service设置成前台服务,8.x退到后台会显示通知栏消息,9.0会立刻显示通知栏消息
startForeground(SERVICE_ID, notification);
}
}
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
bindService(new Intent(this, LocalService.class), serviceConnection, BIND_AUTO_CREATE);
return super.onStartCommand(intent, flags, startId);
}
class MyServiceConnection implements ServiceConnection {
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
}
@Override
public void onServiceDisconnected(ComponentName name) {
Log.e(TAG, "localService 可能被杀死了,拉活");
startService(new Intent(RemoteService.this, LocalService.class));
bindService(new Intent(RemoteService.this, LocalService.class),
serviceConnection, BIND_AUTO_CREATE);
}
}
public static class InnerService extends Service {
@Override
public void onCreate() {
super.onCreate();
Log.e(TAG, "InnerService 服务创建了");
// 让服务变成前台服务
startForeground(16, new Notification());
// 关闭自己
stopSelf();
}
@Nullable
@Override
public IBinder onBind(Intent intent) {
return null;
}
}
}
@SuppressLint("NewApi")
public class DualJobService extends JobService {
public static final String TAG = "DualJobService";
public static void startJob(Context context) {
Log.e(TAG, "startJob");
JobScheduler jobScheduler = (JobScheduler) context.getSystemService(Context.JOB_SCHEDULER_SERVICE);
JobInfo.Builder builder = new JobInfo.Builder(8, new ComponentName(context.getPackageName(),
DualJobService.class.getName())).setPersisted(true);
// 小于7.0
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.N) {
builder.setPeriodic(1000);
} else {
builder.setMinimumLatency(1000);
}
jobScheduler.schedule(builder.build());
}
@Override
public boolean onStartJob(JobParameters params) {
Log.e(TAG, "onStartJob");
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
startJob(this);
}
boolean isLocal = Utils.isRunningService(this, LocalService.class.getName());
boolean isRemote = Utils.isRunningService(this, RemoteService.class.getName());
if (!isLocal || !isRemote) {
startService(new Intent(this, LocalService.class));
startService(new Intent(this, RemoteService.class));
}
return false;
}
@Override
public boolean onStopJob(JobParameters params) {
return false;
}
}
<!--双进程守护-->
<service
android:name=".dual.LocalService"
android:exported="true" />
<service
android:name=".dual.LocalService$InnerService"
android:exported="true" />
<service
android:name=".dual.RemoteService"
android:exported="true"
android:process=":remote" />
<service
android:name=".dual.RemoteService$InnerService"
android:exported="true"
android:process=":remote" />
<service
android:name=".dual.DualJobService"
android:permission="android.permission.BIND_JOB_SERVICE" />
//双进程+job守护
startService(new Intent(this, LocalService.class));
startService(new Intent(this, RemoteService.class));
DualJobService.startJob(this);
6.WorkManager
主要是在AndroidX里使用
//引入
implementation 'androidx.work:work-runtime:2.3.4'
public class KeepLiveWork extends Worker {
public static final String TAG = "KeepLiveWork";
public KeepLiveWork(@NonNull Context context, @NonNull WorkerParameters workerParams) {
super(context, workerParams);
}
@NonNull
@Override
public Result doWork() {
Log.e(TAG, "doWork: ");
MyJobService.startJob(getApplicationContext());
return Result.success();
}
}
@SuppressLint("NewApi")
public class MyJobService extends JobService {
public static final String TAG = "MyJobService";
public static void startJob(Context context) {
Log.e(TAG, "startJob");
JobScheduler jobScheduler = (JobScheduler) context.getSystemService(Context.JOB_SCHEDULER_SERVICE);
JobInfo.Builder builder = new JobInfo.Builder(8, new ComponentName(context.getPackageName(),
MyJobService.class.getName())).setPersisted(true);
// 小于7.0
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.N) {
builder.setPeriodic(1000);
} else {
builder.setMinimumLatency(1000);
}
jobScheduler.schedule(builder.build());
}
@Override
public boolean onStartJob(JobParameters params) {
Log.e(TAG, "onStartJob");
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
startJob(this);
}
return false;
}
@Override
public boolean onStopJob(JobParameters params) {
return false;
}
}
//workManager拉活
OneTimeWorkRequest oneTimeWorkRequest = new OneTimeWorkRequest
.Builder(KeepLiveWork.class)
.setInitialDelay(10, TimeUnit.SECONDS)
.build();
WorkManager.getInstance(this).enqueue(oneTimeWorkRequest);