深入分析 Android Service (完)

文章目录

    • 深入分析 Android Service (完)
    • 1. Service 的生命周期管理
    • 2. Service 的生命周期方法
      • 2.1 onCreate()
      • 2.2 onStartCommand(Intent intent, int flags, int startId)
      • 2.3 onBind(Intent intent)
      • 2.4 onUnbind(Intent intent)
      • 2.5 onRebind(Intent intent)
      • 2.6 onDestroy()
    • 3. Service 重启策略
    • 4. 使用 Service 进行前台任务
    • 5. 实现前台服务示例
      • 5.1 创建前台服务
      • 5.2 启动前台服务
    • 6. Service 的优化和调试
      • 6.1 使用 JobScheduler 替代传统 Service
      • 6.2 使用 WorkManager 处理后台任务
      • 6.3 调试和监控
    • 7. 示例代码汇总
      • 7.1 服务端代码(Messenger)
      • 7.2 客户端代码(Messenger)
      • 7.3 服务端代码(AIDL)
      • 7.4 客户端代码(AIDL)
    • 8. 总结

深入分析 Android Service (完)

1. Service 的生命周期管理

Service 的生命周期管理是确保 Service 能够正确启动、运行、停止和清理资源的关键。理解 Service 的生命周期方法可以帮助开发者更好地管理和优化 Service

2. Service 的生命周期方法

Service 的主要生命周期方法包括:

  1. onCreate()
  2. onStartCommand(Intent intent, int flags, int startId)
  3. onBind(Intent intent)
  4. onUnbind(Intent intent)
  5. onRebind(Intent intent)
  6. onDestroy()

2.1 onCreate()

Service 被创建时调用。通常在这里初始化任何必要的资源。

@Override
public void onCreate() {super.onCreate();// 初始化资源,如线程池、数据库连接等
}

2.2 onStartCommand(Intent intent, int flags, int startId)

在每次通过 startService() 方法启动 Service 时调用。此方法是处理实际业务逻辑的主要入口。返回值决定系统如何在服务因内存不足而被杀死后重启它。

@Override
public int onStartCommand(Intent intent, int flags, int startId) {// 处理业务逻辑,如下载文件、播放音乐等return START_STICKY;  // START_NOT_STICKY, START_REDELIVER_INTENT, or START_STICKY_COMPATIBILITY
}

2.3 onBind(Intent intent)

在客户端绑定到 Service 时调用。返回一个 IBinder 对象,用于与客户端通信。

@Override
public IBinder onBind(Intent intent) {// 返回通信接口return binder;
}

2.4 onUnbind(Intent intent)

在所有客户端都解绑时调用。通常在这里清理与绑定相关的资源。

@Override
public boolean onUnbind(Intent intent) {// 清理绑定相关资源return super.onUnbind(intent);
}

2.5 onRebind(Intent intent)

在之前调用了 onUnbind() 后,新的客户端再次绑定到 Service 时调用。

@Override
public void onRebind(Intent intent) {super.onRebind(intent);
}

2.6 onDestroy()

Service 被销毁前调用。通常在这里清理所有资源。

@Override
public void onDestroy() {super.onDestroy();// 释放资源
}

3. Service 重启策略

onStartCommand 方法的返回值决定了当 Service 因系统内存不足被杀死后,系统如何重启它:

  • START_STICKY:服务被系统终止后会自动重启。Intent 不会保留,意味着数据不会保留,但服务会继续运行。
  • START_NOT_STICKY:服务被系统终止后不会自动重启,除非有新的 Intent。
  • START_REDELIVER_INTENT:服务被系统终止后会自动重启,并重传最后一个 Intent。
  • START_STICKY_COMPATIBILITY:类似于 START_STICKY,但用于兼容低版本。

4. 使用 Service 进行前台任务

为了确保 Service 在后台运行时不被系统终止,可以将 Service 提升为前台服务。这可以通过调用 startForeground() 方法实现,并提供一个持续显示的通知。

5. 实现前台服务示例

5.1 创建前台服务

public class ForegroundService extends Service {@Overridepublic void onCreate() {super.onCreate();Notification notification = createNotification();startForeground(1, notification);}private Notification createNotification() {NotificationManager notificationManager = (NotificationManager) getSystemService(Context.NOTIFICATION_SERVICE);String channelId = "foreground_service_channel";if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) {NotificationChannel channel = new NotificationChannel(channelId, "Foreground Service Channel", NotificationManager.IMPORTANCE_DEFAULT);notificationManager.createNotificationChannel(channel);}NotificationCompat.Builder builder = new NotificationCompat.Builder(this, channelId).setContentTitle("Foreground Service").setContentText("Service is running in the foreground").setSmallIcon(R.drawable.ic_service_icon);return builder.build();}@Overridepublic int onStartCommand(Intent intent, int flags, int startId) {// 处理业务逻辑return START_STICKY;}@Overridepublic IBinder onBind(Intent intent) {return null;}@Overridepublic void onDestroy() {super.onDestroy();stopForeground(true);}
}

5.2 启动前台服务

Activity 中启动前台服务:

Intent intent = new Intent(this, ForegroundService.class);
startService(intent);

6. Service 的优化和调试

为了优化 Service 的性能和可靠性,可以采用以下方法:

6.1 使用 JobScheduler 替代传统 Service

对于需要在特定条件下运行的任务,推荐使用 JobScheduler,它可以根据系统资源和条件优化任务的执行时间。

JobScheduler jobScheduler = (JobScheduler) getSystemService(Context.JOB_SCHEDULER_SERVICE);
JobInfo jobInfo = new JobInfo.Builder(1, new ComponentName(this, MyJobService.class)).setRequiredNetworkType(JobInfo.NETWORK_TYPE_UNMETERED).setRequiresCharging(true).build();
jobScheduler.schedule(jobInfo);

6.2 使用 WorkManager 处理后台任务

WorkManager 提供了一种现代化的任务调度机制,适用于需要保证任务执行的场景。它可以自动处理任务的重试和约束条件。

OneTimeWorkRequest workRequest = new OneTimeWorkRequest.Builder(MyWorker.class).setConstraints(new Constraints.Builder().setRequiresCharging(true).build()).build();
WorkManager.getInstance(this).enqueue(workRequest);

6.3 调试和监控

使用 StrictMode 检测潜在的性能问题,如磁盘读写和网络访问:

if (BuildConfig.DEBUG) {StrictMode.setThreadPolicy(new StrictMode.ThreadPolicy.Builder().detectAll().penaltyLog().build());StrictMode.setVmPolicy(new StrictMode.VmPolicy.Builder().detectAll().penaltyLog().build());
}

利用 Android Studio 的 Profiler 工具监控 Service 的 CPU、内存和网络使用情况,以识别和解决性能瓶颈。

7. 示例代码汇总

7.1 服务端代码(Messenger)

public class MessengerService extends Service {static final int MSG_SAY_HELLO = 1;class IncomingHandler extends Handler {@Overridepublic void handleMessage(Message msg) {switch (msg.what) {case MSG_SAY_HELLO:Toast.makeText(getApplicationContext(), "Hello!", Toast.LENGTH_SHORT).show();break;default:super.handleMessage(msg);}}}final Messenger messenger = new Messenger(new IncomingHandler());@Overridepublic IBinder onBind(Intent intent) {return messenger.getBinder();}
}

7.2 客户端代码(Messenger)

public class MainActivity extends AppCompatActivity {Messenger messenger = null;boolean isBound = false;private ServiceConnection connection = new ServiceConnection() {@Overridepublic void onServiceConnected(ComponentName name, IBinder service) {messenger = new Messenger(service);isBound = true;}@Overridepublic void onServiceDisconnected(ComponentName name) {messenger = null;isBound = false;}};@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);Intent intent = new Intent(this, MessengerService.class);bindService(intent, connection, Context.BIND_AUTO_CREATE);Button sendButton = findViewById(R.id.sendButton);sendButton.setOnClickListener(v -> {if (isBound) {Message msg = Message.obtain(null, MessengerService.MSG_SAY_HELLO, 0, 0);try {messenger.send(msg);} catch (RemoteException e) {e.printStackTrace();}}});}@Overrideprotected void onDestroy() {super.onDestroy();if (isBound) {unbindService(connection);isBound = false;}}
}

7.3 服务端代码(AIDL)

public class MyAidlService extends Service {private final IMyAidlInterface.Stub binder = new IMyAidlInterface.Stub() {@Overridepublic int add(int a, int b) {return a + b;}};@Overridepublic IBinder onBind(Intent intent) {return binder;}
}

7.4 客户端代码(AIDL)

public class MainActivity extends AppCompatActivity {IMyAidlInterface myAidlService = null;boolean isBound = false;private ServiceConnection connection = new ServiceConnection() {@Overridepublic void onServiceConnected(ComponentName name, IBinder service) {myAidlService = IMyAidlInterface.Stub.asInterface(service);isBound = true;}@Overridepublic void onServiceDisconnected(ComponentName name) {myAidlService = null;isBound = false;}};@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);Intent intent = new Intent(this, MyAidlService.class);bindService(intent, connection, Context.BIND_AUTO_CREATE);Button addButton = findViewById(R.id.addButton);addButton.setOnClickListener(v -> {if (isBound) {try {int result = myAidlService.add(5, 3);Toast.makeText(MainActivity.this, "Result: " + result, Toast.LENGTH_SHORT).show();} catch (RemoteException e) {e.printStackTrace();}}});}@Overrideprotected void onDestroy() {super.onDestroy();if (isBound) {unbindService(connection);isBound = false;}}
}

8. 总结

通过合理设计和优化 Android Service,可以确保应用在后台高效、稳定地运行,提供良好的用户体验。希望本系列文章能够帮助开发者更深入地理解 Service 的工作机制和优化策略,为开发高质量的 Android 应用提供参考和指导。

欢迎点赞|关注|收藏|评论,您的肯定是我创作的动力

在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/338335.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

2010-2015 年阿拉斯加北坡苔原植物功能类型连续覆盖图

ABoVE: Tundra Plant Functional Type Continuous-Cover, North Slope, Alaska, 2010-2015 2010-2015 年阿拉斯加北坡苔原植物功能类型连续覆盖图 简介 文件修订日期:2021-08-27 数据集版本: 1 摘要 该数据集以 30 米的分辨率提供了阿拉斯加北坡约 12.5 万平方…

[数据集][图像分类]轮胎纹理识别裂纹和正常轮胎分类数据集1028张2类别

数据集类型:图像分类用,不可用于目标检测无标注文件 数据集格式:仅仅包含jpg图片,每个类别文件夹下面存放着对应图片 图片数量(jpg文件个数):1028 分类类别数:2 类别名称:["cracked","norma…

GIt快速入门(一文学会使用Git)

GIt快速入门 文章目录 GIt快速入门一、为什么要学习Git二、Git的安装1.安装Git2.下载GUI 三、Git的概念1、版本控制2、集中式控制3、分布式控制4、多人协作开发1.并行开发2.分支管理3.冲突解决4.代码审查5.分布式特性 四、Git客户端操作1.界面介绍2.提交操作3.创建分支4.合并分…

前端框架安全防范

前端框架安全防范 在现代Web开发中,前端框架如Angular和React已经成为构建复杂单页面应用(SPA)的主流工具。然而,随着应用复杂度的增加,安全问题也变得越来越重要。本文将介绍如何在使用Angular和React框架时&#xf…

HashMap在Go与Java的底层实现与区别

在Java中 在Java中hash表的底层数据结构与扩容等已经是面试集合类问题中几乎必问的点了。网上有对源码的解析已经非常详细了我们这里还是说说其底层实现。 基础架构 public class HashMap<K,V> extends AbstractMap<K,V> implements Map<K,V>, Cloneable,…

通过LabVIEW提升生产设备自动化水平

现代制造业对生产设备的自动化水平提出了越来越高的要求。使用LabVIEW这一强大的图形化编程环境&#xff0c;可以显著提升生产设备的自动化程度&#xff0c;改善生产效率和产品质量。本文将详细分析如何通过LabVIEW改善生产设备的自动化水平&#xff0c;并提供具体的实施策略与…

[数据集][图像分类]骨关节炎严重程度分类数据集14038张4分类

数据集类型&#xff1a;图像分类用&#xff0c;不可用于目标检测无标注文件 数据集格式&#xff1a;仅仅包含jpg图片&#xff0c;每个类别文件夹下面存放着对应图片 图片数量(jpg文件个数)&#xff1a;14038 分类类别数&#xff1a;4 类别名称:[“grade0”,“grade2”,“grade3…

如何定义“智慧校园”这个概念

在信息爆炸的时代&#xff0c;教育面临着前所未有的挑战&#xff1a;如何让每个学生在海量知识中找到属于自己的路径&#xff1f;如何让教师的智慧与科技的力量相得益彰&#xff1f;如何让校园成为培养创新思维的摇篮&#xff1f;智慧校园&#xff0c;这一概念的提出&#xff0…

基于 Redis 实现分布式锁的全过程

前言 这一篇文章拖了有点久&#xff0c;虽然在项目中使用分布式锁的频率比较高&#xff0c;但整理成文章发布出来还是花了一点时间。在一些移动端、用户量大的互联网项目中&#xff0c;经常会使用到 Redis 分布式锁作为控制访问高并发的工具。 一、关于分布式锁 总结&#x…

计算机图形学入门05:投影变换

1.投影变换 上一章已经介绍了投影变换&#xff0c;就是将三维图像投影到二维平面上&#xff0c;而投影变换又分为正交投影(Orthographic Projection)和透视投影(Perspective Projection)。如下图&#xff1a; 正交投影 没有近大远小的现象&#xff0c;无论图形与视点距离是远是…

机器学习学习

机器学习类型(按学习方式分):监督学习、半监督学习、无监督学习、强化学习; 通过已知标签训练集训练模型,使用模型及逆行预测、测试; 向量表示法,其中每一维对应一个特征(feature)或者称为属性,记为[x1,x2,...,xn] 特征值、特征、标签,共同完成训练集的数据填充,最…

设计模式基础知识点(七大原则、UML类图)

Java设计模式&#xff08;设计模式七大原则、UML类图&#xff09; 设计模式的目的设计模式七大原则单一职能原则&#xff08;SingleResponsibility&#xff09;接口隔离原则&#xff08;InterfaceSegreation&#xff09;依赖倒转原则&#xff08;DependenceInversion&#xff0…

Python 关于字符串格式化

在Python中&#xff0c;字符串格式化有以下几种方法&#xff1a; 1.可以使用字符串的str.center(width), str.ljust(width), 和 str.rjust(width)方法来实现字符串的居中、左对齐和右对齐操作。 居中对齐&#xff1a; text "Python" centered_text text.center(10…

pytest-sugar插件:对自动化测试用例加入进度条

摘要 在自动化测试过程中&#xff0c;测试进度的可视化对于开发者和测试工程师来说非常重要。本文将介绍如何使用pytest-sugar插件来为pytest测试用例添加进度条&#xff0c;从而提升测试的可读性和用户体验。 1. 引言 自动化测试是软件开发过程中不可或缺的一部分&#xff…

Django Celery技术详解

文章目录 简介安装和配置创建并调度任务启动Celery Worker在视图中调用异步任务拓展功能 简介 Django Celery 是一个为Django应用程序提供异步任务处理能力的强大工具。它通过与消息代理&#xff08;如RabbitMQ、Redis&#xff09;集成&#xff0c;可以轻松地处理需要长时间运…

Windows11 wsl2编译Android14 使用ASfP Debug windows上启动的模拟器

wsl2的安装和配置 安装&#xff1a; 直接百度搜索最新的wsl2安装教程即可&#xff0c;官网&#xff1a;https://learn.microsoft.com/zh-cn/windows/wsl/install 1. 启用适用于 Linux 的 Windows 子系统(以管理员身份打开 PowerShell 并运行) Enable-WindowsOptionalFeature…

C++ | Leetcode C++题解之第125题验证回文串

题目&#xff1a; 题解&#xff1a; class Solution { public:bool isPalindrome(string s) {int n s.size();int left 0, right n - 1;while (left < right) {while (left < right && !isalnum(s[left])) {left;}while (left < right && !isalnu…

fyne apptab布局

fyne apptab布局 AppTabs 容器允许用户在不同的内容面板之间切换。标签要么只是文本&#xff0c;要么是文本和一个图标。建议不要混合一些有图标的标签和一些没有图标的标签。 package mainimport ("fyne.io/fyne/v2/app""fyne.io/fyne/v2/container"//&…

全国产飞腾模块麒麟信安操作系统安全漏洞

1、背景介绍 目前在全国产飞腾模块上部署了麒麟信安操作系统&#xff0c;经第三方机构检测存在以下漏洞 操作系统版本为 内核版本为 openssh版本为 2、openssh CBC模式漏洞解决 首先查看ssh加密信息 nmap --script "ssh2*" 127.0.0.1 | grep -i cbc 可以通过修改/…

Python | Leetcode Python题解之第125题验证回文串

题目&#xff1a; 题解&#xff1a; class Solution:def isPalindrome(self, s: str) -> bool:n len(s)left, right 0, n - 1while left < right:while left < right and not s[left].isalnum():left 1while left < right and not s[right].isalnum():right - …