深入分析 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.mzph.cn/news/846100.shtml

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

相关文章

子集树与排列树的构造

排列树的构造: 无重复画法:一条线前面出现的不再出现。 有重复画法:一条线前面出现的不再出现,如果仅仅只是相似可以出现;兄弟不能相似。 目标函数是:cnt 总元素个数分支策略是全遍历,不过存…

二级指针简单介绍

我们之前学习的&#xff1a;变量的地址是存入指针变量中的&#xff0c;然而指针变量也是变量&#xff0c;是变量就有地址&#xff0c;那么指针变量的地址存放在哪里 &#xff1f; 这也就是二级指针 #include<stdio.h> int main() {int a10;int*p&a;int**pp&p;re…

微软如何打造数字零售力航母系列科普12 - 使用Microsoft Fabric将客户数据带入人工智能时代

【世界上充斥着数据&#xff0c;在过去的2年里&#xff0c;我们都看到了人工智能如何有潜力彻底改变我们的日常业务。人们对利用生成性人工智能体验的力量的需求越来越大&#xff0c;但这样做需要一个干净的数据庄园&#xff0c;而且可能会因为各种技术堆栈、分散的团队和无处不…

如何让Google快速收录?

要让Google快速收录你的网站&#xff0c;可以考虑使用GSI服务&#xff0c;这是一种专门设计来加速网站被Google搜索引擎收录的服务&#xff0c;下面详细解释GSI服务的基本原理和具体好处&#xff1a; GSI服务通过一种名为GPC爬虫池的系统实现&#xff0c;这个系统是基于对Goog…

eBPF可观测之网络流量控制和管理traffic control浅尝

目录 工程背景 环境准备 安装工具​​​ 安装依赖包 安装C依赖库 操作步骤 目录结构 代码展示 效果展示 拓展提升 工程背景 首先发表一个"暴论" eBPF在可观测方面的应用&#xff0c;就是各种google。 不需要学习内核&#xff0c;只要掌握ebpf开发套路。…

什么是TLAB?

这个得从内存申请说起。 一般而言生成对象需要向堆中的新生代申请内存空间&#xff0c;而堆又是全局共享的&#xff0c;像新生代内存又是规整的&#xff0c;是通过一个指针来划分的。 内存是紧凑的&#xff0c;新对象创建指针就右移对象大小size即可&#xff0c;这叫指针加法…

怎么通过互联网远程控制电脑?

远程访问又称为网络远程控制&#xff0c;它使用户能够通过互联网连接两台设备以解决问题。进行控制的电脑称为控制端&#xff0c;被控制的电脑则称为被控端。在远程访问过程中&#xff0c;控制端电脑掌握整个连接的操作。远程控制软件会捕获被控端电脑的操作&#xff0c;并在主…

绘画智能体分享

这是您请求的故宫雪景图&#xff0c;角落有一只可爱的胖猫&#xff0c;采用了水墨画风格&#xff0c;类似于张大千的作品。希望您喜欢这幅画&#xff01; &#x1f3a8; 选项 1【转变风格】——将这幅画转变为梵高的后印象派风格&#xff0c;增添一些梵高特有的笔触和色彩。 &…

Superset二次开发之Github项目推送到GitLab仓库

以下是从GitHub克隆Superset项目并将其推送到GitLab的详细操作步骤 lab 地址: xxx lab 配置: 生成SSH密钥 ssh-keygen -t rsa -b 4096 -C "邮箱地址" 默认情况下密钥会生成在~/.ssh (/c/Users/Administrator/.ssh/id_rsa)目录下。 公钥添加到GitLab: 打开公钥文件…

短时间内如何顺利通过 Java 面试?

今天我们来探讨一个重要的话题&#xff1a;短时间内如何顺利通过 Java 面试&#xff1f; 在此之前&#xff0c;我正在精心编写一套完全面向小白的 Java 自学教程&#xff0c;我相信这套教程会非常适合正在努力提升的你。教程里面涵盖了丰富全面的编程教学内容、详细生动的视频…

上网行为监控软件大盘点:好用的上网行为管理软件一览

这是后台私信来的客户发来的图片&#xff0c;他们老板想要给自己公司安装上网行为监控软件。 可见&#xff0c;企业对员工上网行为的监控与管理已经是很多企业都开始重视的问题。 上网行为管理软件能够帮助企业确保网络资源得到合理分配&#xff0c;提升工作效率&#xff0c;…

ch5链路层和局域网

回顾TCP/IP参考模型&#xff0c;明确链路层和物理层在整个模型中的地位&#xff0c;简要提出链路层要解决的问题是单段链路的数据传输&#xff0c;物理层解决的是数字信号与电气信号之间的相互转换。 链路层概述 节点&#xff1a;主机和路由器(包括网桥和交换机) 链路&#xf…

QT 信号和槽教程,窗体和控件对象之间的沟通一般都使用信号和槽

Qt的信号和槽&#xff08;Signals and Slots&#xff09;机制是一种强大的对象间通信方式&#xff0c;它允许对象在完全解耦的情况下相互通信。以下是关于Qt信号和槽的简明教程&#xff1a; 基本概念 信号&#xff08;Signal&#xff09;&#xff1a;信号是由Qt对象发出的通知…

(2024,LoRA,全量微调,低秩,强正则化,缓解遗忘,多样性)LoRA 学习更少,遗忘更少

LoRA Learns Less and Forgets Less 公和众和号&#xff1a;EDPJ&#xff08;进 Q 交流群&#xff1a;922230617 或加 VX&#xff1a;CV_EDPJ 进 V 交流群&#xff09; 目录 0. 摘要 1. 引言 2. 背景 3. 实验设置 3.2 使用编码和数学基准测试来衡量学习&#xff08;目标域…

jpeg编码学习

正点原子stm32教程提到过jpeg解码库libjpeg&#xff0c;但是没有提到jpeg编码&#xff0c;我也好奇jpeg编码怎么实现&#xff0c;用代码怎么生成jpeg文件的。所以最近学习了jpeg编码&#xff0c;在这里做记录。 参考文章 jpeg图片格式详解 https://blog.csdn.net/yun_hen/art…

一文了解-电子公司组织架构

以下仅为一般研发设计公司的岗位&#xff0c; 不包括工厂。 总经理 柏柏有话说: 不多说&#xff0c;人中龙凤、群山俯首、众神膜拜&#xff01; 项目总监 产品总监 软件总监 测试总监 销售总监 人事总监 财务总监 采购总监 柏柏有话说: 这么说吧&#xff0c; 柏柏的前同龄同事…

共计3万字!从零开始创建一个小规模的稳定扩散模型!

节前&#xff0c;我们星球组织了一场算法岗技术&面试讨论会&#xff0c;邀请了一些互联网大厂朋友、参加社招和校招面试的同学。 针对算法岗技术趋势、大模型落地项目经验分享、新手如何入门算法岗、该如何准备、面试常考点分享等热门话题进行了深入的讨论。 合集&#x…

MYSQL数据库细节详细分析

MYSQL数据库的数据类型(一般只需要用到这些) 整型类型&#xff1a;用于存储整数值&#xff0c;可以选择不同的大小范围来适应特定的整数值。 TINYINTSMALLINTMEDIUMINTINTBIGINT 浮点型类型&#xff1a;用于存储带有小数部分的数值&#xff0c;提供了单精度&#xff08;FLOA…

YOLOv5改进 | 卷积模块 | 将Conv替换为轻量化的GSConv【原理 + 完整代码】

&#x1f4a1;&#x1f4a1;&#x1f4a1;本专栏所有程序均经过测试&#xff0c;可成功执行&#x1f4a1;&#x1f4a1;&#x1f4a1; 目标检测是计算机视觉中一个重要的下游任务。对于边缘盒子的计算平台来说&#xff0c;一个大型模型很难实现实时检测的要求。而且&#xff0…

全志H616 通过Cedrus和v4l2_request API实现硬件编解码加速(香橙派zero2)

编译安装或加载cedrus驱动模块&#xff0c;加载v4l2-mem2mem Sunxi-Cedrus 致力于为全志 SoC 提供硬件加速的视频解码和编码支持&#xff0c;并将其引入主线 Linux 内核。此外&#xff0c;还为典型的基于 GNU/Linux 的系统提供了与内核驱动程序接口的其他用户空间组件。 Sunx…