服务(Service)全解析(六)--避免被系统回收的Service

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
简介: MainActivity如下: package cc.c;import java.util.List;import android.os.

MainActivity如下:

package cc.c;

import java.util.List;
import android.os.Bundle;
import android.os.Process;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
import android.app.Activity;
import android.app.ActivityManager;
import android.app.ActivityManager.RunningAppProcessInfo;
import android.app.ActivityManager.RunningTaskInfo;
import android.content.ComponentName;
import android.content.Context;
/**
 * Demo描述:
 * 实现防止被系统回收的Service.
 * 在某些情况下Service会被系统kill掉.在网上找了不少方法,虽然都号称
 * 可以避免被系统回收,但貌似都不够可靠.
 * 在此换一种方式来实现,思路:
 * 利用AlarmManager定时间隔(比如5S,20S)来不断的startService().
 * (1)如果该服务被回收,那么会调用Service的onCreate()和onStart()
 * (2)如果未该服务被回收,则只会调用Service的onStart()
 * 利用该方式曲线实现防止被系统回收Service
 * 
 * 测试环境:
 * Android:2.3.6 + ME525+
 * 
 * 测试方法:
 * 1 部署应用到设备
 * 2 重启设备
 * 3 设备重启后,点击进入该应用.点击界面上的Button
 * 4 查看Log
 * 
 * 
 * 注意事项:
 * 1 4.0及其以上请注意开机广播监听的实现
 * 2 注意权限
 *   <uses-permission android:name="android.permission.KILL_BACKGROUND_PROCESSES" />
 *   <uses-permission android:name="android.permission.GET_TASKS"/>  
 *   
 * 备注说明:
 * 1 在杀死某进程时有时需要判断该进程所属的Activity是否正在屏幕上显示
 *   该方法请参见getTopActivityNameAndProcessName()
 * 2 若有其余的方法,请留言指教.多谢.
 * 
 */
public class MainActivity extends Activity {
	private Button mButton;
	private long lastTime = 0;
	private Context mContext;

	@Override
	protected void onCreate(Bundle savedInstanceState) {
		super.onCreate(savedInstanceState);
		setContentView(R.layout.main);
		init();
	}

	private void init() {
		mContext = this;
		lastTime = System.currentTimeMillis();
		mButton = (Button) findViewById(R.id.button);
		mButton.setOnClickListener(new OnClickListener() {
			@Override
			public void onClick(View view) {
				if ((System.currentTimeMillis() - lastTime) > 2000) {
					lastTime = System.currentTimeMillis();
					cleanMemory(mContext);
				}

			}
		});
	}

	//模拟服务被系统回收
	private void cleanMemory(Context context) {
		RunningAppProcessInfo runningAppProcessInfo = null;
		ActivityManager activityManager =
		(ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE);
		List<RunningAppProcessInfo> runningAppProcessInfoList = 
		activityManager.getRunningAppProcesses();
		if (runningAppProcessInfoList != null) {
			for (int i = 0; i < runningAppProcessInfoList.size(); ++i) {
				runningAppProcessInfo = runningAppProcessInfoList.get(i);
				String processName = runningAppProcessInfo.processName;
				if (processName.startsWith("cc.c")) {
					System.out.println("------>杀掉本应用");
					int pid = runningAppProcessInfo.pid;
					Process.killProcess(pid);
				}
			}
		}

	}
	
	//获取栈顶Activity及其所属进程
	public static String getTopActivityNameAndProcessName(Context context){
		String processName=null;
		String topActivityName=null;
		 ActivityManager activityManager =
		(ActivityManager)(context.getSystemService(android.content.Context.ACTIVITY_SERVICE )) ;
	     List<RunningTaskInfo> runningTaskInfos = activityManager.getRunningTasks(1) ;
	     if(runningTaskInfos != null){
	    	 ComponentName f=runningTaskInfos.get(0).topActivity;
	    	 String topActivityClassName=f.getClassName();
	    	 String temp[]=topActivityClassName.split("\\.");
	    	 //栈顶Activity的名称
	    	 topActivityName=temp[temp.length-1];
	    	 int index=topActivityClassName.lastIndexOf(".");
	    	//栈顶Activity所属进程的名称
	    	 processName=topActivityClassName.substring(0, index);
	    	 System.out.println("---->topActivityName="+topActivityName+",processName="+processName);
	    	 
	     }
	     return topActivityName+","+processName;
	}

}

BootCompletedBroadcastReceiver如下:

package cc.c;

import android.app.AlarmManager;
import android.app.PendingIntent;
import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
import android.os.SystemClock;
//监听开机广播
public class BootCompletedBroadcastReceiver extends BroadcastReceiver {
	@Override
	public void onReceive(Context context, Intent intent) {
		//接收到开机广播
		if(intent.getAction().equals(Intent.ACTION_BOOT_COMPLETED)){
			sendCheckServiceBroadcast(context);
		}

	}
	
	//发送广播
	private void sendCheckServiceBroadcast(Context context) {
		Intent intent = new Intent(context, CheckServiceBroadcastReceiver.class);
		intent.setAction("com.cn.checkServicebroadcasreceiver");
		PendingIntent pendingIntent = PendingIntent.getBroadcast(context, 0, intent, 0);
		long firstime = SystemClock.elapsedRealtime();
		AlarmManager alarmManager = (AlarmManager) context.getSystemService(Context.ALARM_SERVICE);
		//每间隔20秒发送广播startService
		alarmManager.setRepeating(AlarmManager.ELAPSED_REALTIME_WAKEUP, firstime,20 * 1000, pendingIntent);
	}
	
}

CheckServiceBroadcastReceiver如下:

package cc.c;

import android.content.BroadcastReceiver;
import android.content.Context;
import android.content.Intent;
//检查Service
public class CheckServiceBroadcastReceiver extends BroadcastReceiver {
	@Override
	public void onReceive(Context context, Intent intent) {
		if (intent.getAction().equals("com.cn.checkServicebroadcasreceiver")) {
			Intent serviceIntent = new Intent();
			serviceIntent.setAction("cn.com.servicesubclass");
			context.startService(serviceIntent);
		}

	}
}

ServiceSubclass如下:

package cc.c;

import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
//执行业务逻辑的Service
public class ServiceSubclass extends Service {

	@Override
	public IBinder onBind(Intent arg0) {
		return null;
	}

	@Override
	public void onCreate() {
		super.onCreate();
		System.out.println("------>service onCreate()");
	}
	
	@Override
	public void onStart(Intent intent, int startId) {
		super.onStart(intent, startId);
		System.out.println("------>service onStart()");
	}
	

}

main.xml如下:

<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
   >

    <Button
        android:id="@+id/button"
        android:layout_width="wrap_content"
        android:layout_height="wrap_content"
        android:layout_centerInParent="true"
        android:text="杀掉进程" />

</RelativeLayout>

AndroidManifest.xml如下:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
    package="cc.c"
    android:versionCode="1"
    android:versionName="1.0" >

    <uses-sdk
        android:minSdkVersion="10"
        android:targetSdkVersion="10" />
    
    <!-- 杀死进程的权限 -->
    <uses-permission android:name="android.permission.KILL_BACKGROUND_PROCESSES" />
     <!-- 获取Task的权限 -->
    <uses-permission android:name="android.permission.GET_TASKS"/>  

    <application
        android:allowBackup="true"
        android:icon="@drawable/ic_launcher"
        android:label="@string/app_name"
        android:theme="@style/AppTheme" >
        <activity
            android:name="cc.c.MainActivity"
            android:label="@string/app_name" >
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />

                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>

        <!-- 注册开机广播接收者 -->
        <receiver android:name=".BootCompletedBroadcastReceiver" >
            <intent-filter>
                <action android:name="android.intent.action.BOOT_COMPLETED" />
            </intent-filter>
        </receiver>
        
         <!-- 注册广播接收者-->
        <receiver android:name=".CheckServiceBroadcastReceiver" >
            <intent-filter>
                <action android:name="com.cn.checkServicebroadcasreceiver" />
            </intent-filter>
        </receiver>
        
        <!-- 注册服务 -->
        <service android:name=".ServiceSubclass">
            <intent-filter>
                <action android:name="cn.com.servicesubclass"/>
            </intent-filter>
        </service>
        
        
    </application>

</manifest>


相关文章
|
3月前
|
存储 缓存 算法
分布式锁服务深度解析:以Apache Flink的Checkpointing机制为例
【10月更文挑战第7天】在分布式系统中,多个进程或节点可能需要同时访问和操作共享资源。为了确保数据的一致性和系统的稳定性,我们需要一种机制来协调这些进程或节点的访问,避免并发冲突和竞态条件。分布式锁服务正是为此而生的一种解决方案。它通过在网络环境中实现锁机制,确保同一时间只有一个进程或节点能够访问和操作共享资源。
104 3
|
6天前
|
存储 监控 算法
企业内网监控系统中基于哈希表的 C# 算法解析
在企业内网监控系统中,哈希表作为一种高效的数据结构,能够快速处理大量网络连接和用户操作记录,确保网络安全与效率。通过C#代码示例展示了如何使用哈希表存储和管理用户的登录时间、访问IP及操作行为等信息,实现快速的查找、插入和删除操作。哈希表的应用显著提升了系统的实时性和准确性,尽管存在哈希冲突等问题,但通过合理设计哈希函数和冲突解决策略,可以确保系统稳定运行,为企业提供有力的安全保障。
|
27天前
|
安全 前端开发 Android开发
探索移动应用与系统:从开发到操作系统的深度解析
在数字化时代的浪潮中,移动应用和操作系统成为了我们日常生活的重要组成部分。本文将深入探讨移动应用的开发流程、关键技术和最佳实践,同时分析移动操作系统的核心功能、架构和安全性。通过实际案例和代码示例,我们将揭示如何构建高效、安全且用户友好的移动应用,并理解不同操作系统之间的差异及其对应用开发的影响。无论你是开发者还是对移动技术感兴趣的读者,这篇文章都将为你提供宝贵的见解和知识。
|
1月前
|
负载均衡 网络协议 算法
Docker容器环境中服务发现与负载均衡的技术与方法,涵盖环境变量、DNS、集中式服务发现系统等方式
本文探讨了Docker容器环境中服务发现与负载均衡的技术与方法,涵盖环境变量、DNS、集中式服务发现系统等方式,以及软件负载均衡器、云服务负载均衡、容器编排工具等实现手段,强调两者结合的重要性及面临挑战的应对措施。
74 3
|
10天前
|
安全 搜索推荐 数据挖掘
陪玩系统源码开发流程解析,成品陪玩系统源码的优点
我们自主开发的多客陪玩系统源码,整合了市面上主流陪玩APP功能,支持二次开发。该系统适用于线上游戏陪玩、语音视频聊天、心理咨询等场景,提供用户注册管理、陪玩者资料库、预约匹配、实时通讯、支付结算、安全隐私保护、客户服务及数据分析等功能,打造综合性社交平台。随着互联网技术发展,陪玩系统正成为游戏爱好者的新宠,改变游戏体验并带来新的商业模式。
|
2月前
|
机器学习/深度学习 人工智能 数据处理
【AI系统】NV Switch 深度解析
英伟达的NVSwitch技术是高性能计算领域的重大突破,旨在解决多GPU系统中数据传输的瓶颈问题。通过提供比PCIe高10倍的带宽,NVLink实现了GPU间的直接数据交换,减少了延迟,提高了吞吐量。NVSwitch则进一步推动了这一技术的发展,支持更多NVLink接口,实现无阻塞的全互联GPU系统,极大提升了数据交换效率和系统灵活性,为构建强大的计算集群奠定了基础。
75 3
|
2月前
|
网络协议 网络安全 网络虚拟化
本文介绍了十个重要的网络技术术语,包括IP地址、子网掩码、域名系统(DNS)、防火墙、虚拟专用网络(VPN)、路由器、交换机、超文本传输协议(HTTP)、传输控制协议/网际协议(TCP/IP)和云计算
本文介绍了十个重要的网络技术术语,包括IP地址、子网掩码、域名系统(DNS)、防火墙、虚拟专用网络(VPN)、路由器、交换机、超文本传输协议(HTTP)、传输控制协议/网际协议(TCP/IP)和云计算。通过这些术语的详细解释,帮助读者更好地理解和应用网络技术,应对数字化时代的挑战和机遇。
116 3
|
2月前
|
前端开发 Android开发 UED
移动应用与系统:从开发到优化的全面解析####
本文深入探讨了移动应用开发的全过程,从最初的构思到最终的发布,并详细阐述了移动操作系统对应用性能和用户体验的影响。通过分析当前主流移动操作系统的特性及差异,本文旨在为开发者提供一套全面的开发与优化指南,确保应用在不同平台上均能实现最佳表现。 ####
31 0
|
2月前
|
域名解析 缓存 网络协议
浏览器中输入URL返回页面过程(超级详细)、DNS域名解析服务,TCP三次握手、四次挥手
浏览器中输入URL返回页面过程(超级详细)、DNS域名解析服务,TCP三次握手、四次挥手
|
2月前
|
安全 测试技术 数据安全/隐私保护
原生鸿蒙应用市场开发者服务的技术解析:从集成到应用发布的完整体验
原生鸿蒙应用市场开发者服务的技术解析:从集成到应用发布的完整体验

推荐镜像

更多