背景

项目中对于一些并不复杂的耗时操作,比如计算,不频繁操作数据库等,因为没必要使用线程池,所以之前项目会直接使用new Thread的方式,时间一长,回头再看,原来new Thread之处已经很多了,这样带来了一些问题:

  1. 不断的new Thread,损耗性能。
  2. 在有生命周期的类(Activity或者Fragment或者Service)中,有可能出现内存泄漏。
  3. 缺乏统一管理,维护不方便。

分析

问题抛出了,就想办法解决吧。对于任务量小,操作不那么频繁的,我们只需要放在一个后台线程中即能满足要求,思路是只需要分门别类就可以了,对于操作数据库的,都使用data线程,对于计算相关的,都使用background线程,这样整个项目只需要维护几个固定后台线程。

注意:

  1. data 和background都是线程的名字,可以自己添加或者定义。
  2. 因为项目中已经维护了自己封装的线程池,所有涉及到下载,请求等大量I/O操作会使用线程池。

解决方案

自己封装了ThreadManager。

  1. public class ThreadManager {
  2. /** 主线程 */
  3. public static final int THREAD_UI = 0;
  4. /** background线程,用于一般的耗时操作 */
  5. public static final int THREAD_BACKGROUND = 1;
  6. /** data线程,用于数据库操作 */
  7. public static final int THREAD_DATA = 2;
  8. private static final int THREAD_SIZE = 3;
  9. /** 线程信息数组 */
  10. private static final Handler[] HANDLER_LIST = new Handler[THREAD_SIZE];
  11. private static final String[] THREAD_NAME_LIST = {
  12. "thread_ui",
  13. "thread_background",
  14. "thread_data"
  15. };
  16. private ThreadManager() {
  17. HANDLER_LIST[THREAD_UI] = new Handler();
  18. }
  19. private static class ThreadManagerHolder {
  20. private static ThreadManager sManager = new ThreadManager();
  21. }
  22. public static ThreadManager getManager() {
  23. return ThreadManagerHolder.sManager;
  24. }
  25. /**
  26. * 派发任务
  27. *
  28. * @param index 线程类型
  29. */
  30. public void post(int index, Runnable r) {
  31. postDelayed(index, r, 0);
  32. }
  33. /**
  34. * 延迟派发任务
  35. *
  36. * @param index 线程类型
  37. */
  38. public void postDelayed(int index, Runnable r, long delayMillis) {
  39. Handler handler = getHandler(index);
  40. handler.postDelayed(r, delayMillis);
  41. }
  42. /**
  43. * 删除任务
  44. *
  45. * @param index 线程类型
  46. */
  47. public void removeCallbacks(int index, Runnable r) {
  48. Handler handler = getHandler(index);
  49. handler.removeCallbacks(r);
  50. }
  51. /**
  52. * 获取线程Handler
  53. *
  54. * @param index 线程类型
  55. */
  56. public Handler getHandler(int index) {
  57. if (index < 0 || index >= THREAD_SIZE) {
  58. throw new InvalidParameterException();
  59. }
  60. if (HANDLER_LIST[index] == null) {
  61. synchronized (HANDLER_LIST) {
  62. if (HANDLER_LIST[index] == null) {
  63. HandlerThread thread = new HandlerThread(THREAD_NAME_LIST[index]);
  64. if (index != THREAD_UI) {
  65. //优先级要低于主线程
  66. thread.setPriority(Thread.MIN_PRIORITY);
  67. }
  68. thread.start();
  69. Handler handler = new Handler(thread.getLooper());
  70. HANDLER_LIST[index] = handler;
  71. }
  72. }
  73. }
  74. return HANDLER_LIST[index];
  75. }
  76. /**
  77. * 判断是否运行在当前线程
  78. *
  79. * @param index 线程类型
  80. * @return true yes
  81. */
  82. public boolean runningOnCurrent(int index) {
  83. return getHandler(index).getLooper() == Looper.myLooper();
  84. }
  85. }

注释一目了然,不做解释。

来自:

背景

项目中对于一些并不复杂的耗时操作,比如计算,不频繁操作数据库等,因为没必要使用线程池,所以之前项目会直接使用new Thread的方式,时间一长,回头再看,原来new Thread之处已经很多了,这样带来了一些问题:

  1. 不断的new Thread,损耗性能。
  2. 在有生命周期的类(Activity或者Fragment或者Service)中,有可能出现内存泄漏。
  3. 缺乏统一管理,维护不方便。

分析

问题抛出了,就想办法解决吧。对于任务量小,操作不那么频繁的,我们只需要放在一个后台线程中即能满足要求,思路是只需要分门别类就可以了,对于操作数据库的,都使用data线程,对于计算相关的,都使用background线程,这样整个项目只需要维护几个固定后台线程。

注意:

  1. data 和background都是线程的名字,可以自己添加或者定义。
  2. 因为项目中已经维护了自己封装的线程池,所有涉及到下载,请求等大量I/O操作会使用线程池。

解决方案

自己封装了ThreadManager。

  1. public class ThreadManager {
  2. /** 主线程 */
  3. public static final int THREAD_UI = 0;
  4. /** background线程,用于一般的耗时操作 */
  5. public static final int THREAD_BACKGROUND = 1;
  6. /** data线程,用于数据库操作 */
  7. public static final int THREAD_DATA = 2;
  8. private static final int THREAD_SIZE = 3;
  9. /** 线程信息数组 */
  10. private static final Handler[] HANDLER_LIST = new Handler[THREAD_SIZE];
  11. private static final String[] THREAD_NAME_LIST = {
  12. "thread_ui",
  13. "thread_background",
  14. "thread_data"
  15. };
  16. private ThreadManager() {
  17. HANDLER_LIST[THREAD_UI] = new Handler();
  18. }
  19. private static class ThreadManagerHolder {
  20. private static ThreadManager sManager = new ThreadManager();
  21. }
  22. public static ThreadManager getManager() {
  23. return ThreadManagerHolder.sManager;
  24. }
  25. /**
  26. * 派发任务
  27. *
  28. * @param index 线程类型
  29. */
  30. public void post(int index, Runnable r) {
  31. postDelayed(index, r, 0);
  32. }
  33. /**
  34. * 延迟派发任务
  35. *
  36. * @param index 线程类型
  37. */
  38. public void postDelayed(int index, Runnable r, long delayMillis) {
  39. Handler handler = getHandler(index);
  40. handler.postDelayed(r, delayMillis);
  41. }
  42. /**
  43. * 删除任务
  44. *
  45. * @param index 线程类型
  46. */
  47. public void removeCallbacks(int index, Runnable r) {
  48. Handler handler = getHandler(index);
  49. handler.removeCallbacks(r);
  50. }
  51. /**
  52. * 获取线程Handler
  53. *
  54. * @param index 线程类型
  55. */
  56. public Handler getHandler(int index) {
  57. if (index < 0 || index >= THREAD_SIZE) {
  58. throw new InvalidParameterException();
  59. }
  60. if (HANDLER_LIST[index] == null) {
  61. synchronized (HANDLER_LIST) {
  62. if (HANDLER_LIST[index] == null) {
  63. HandlerThread thread = new HandlerThread(THREAD_NAME_LIST[index]);
  64. if (index != THREAD_UI) {
  65. //优先级要低于主线程
  66. thread.setPriority(Thread.MIN_PRIORITY);
  67. }
  68. thread.start();
  69. Handler handler = new Handler(thread.getLooper());
  70. HANDLER_LIST[index] = handler;
  71. }
  72. }
  73. }
  74. return HANDLER_LIST[index];
  75. }
  76. /**
  77. * 判断是否运行在当前线程
  78. *
  79. * @param index 线程类型
  80. * @return true yes
  81. */
  82. public boolean runningOnCurrent(int index) {
  83. return getHandler(index).getLooper() == Looper.myLooper();
  84. }
  85. }

注释一目了然,不做解释。

使用方法

注意:

在绑定生命周期的api中使用,需要主动removeCallBacks,防止内存泄漏。

结语

以上就是关于项目中一些固定线程的封装。

水平有限,如有不对之处,欢迎指出。

使用方法

注意:

在绑定生命周期的api中使用,需要主动removeCallBacks,防止内存泄漏。

结语

以上就是关于项目中一些固定线程的封装。

水平有限,如有不对之处,欢迎指出。

Android中关于项目中对Thread的管理(不是线程池)的更多相关文章

  1. Android异步处理一:使用Thread+Handler实现非UI线程更新UI界面

    Android应用的开发过程中需要把繁重的任务(IO,网络连接等)放到其他线程中异步执行,达到不阻塞UI的效果. 下面将由浅入深介绍Android进行异步处理的实现方法和系统底层的实现原理. 本文介绍 ...

  2. Android线程管理之ExecutorService线程池

    前言: 上篇学习了线程Thread的使用,今天来学习一下线程池ExecutorService. 线程管理相关文章地址: Android线程管理之Thread使用总结 Android线程管理之Execu ...

  3. 通过Thread Pool Executor类解析线程池执行任务的核心流程

    摘要:ThreadPoolExecutor是Java线程池中最核心的类之一,它能够保证线程池按照正常的业务逻辑执行任务,并通过原子方式更新线程池每个阶段的状态. 本文分享自华为云社区<[高并发] ...

  4. Android Studio 在项目中引用第三方jar包

    在Android Studio项目中引用第三方jar包的方法: 步骤: 1.在build.gradle文件中添加如下代码: 备注:要添加在Android作用域下 sourceSets { main { ...

  5. android直接读取项目中的sqlite数据库

    最近项目中要实现android读取sqlite数据库文件,在这里先做一个英汉字典的例子.主要是输入英语到数据库中查询相应的汉语意思,将其答案输出.数据库采用sqlite3. 如图: 实现过程完全是按照 ...

  6. Android Studio修改项目中整体包名

    莫名的需求,要把之前的apk分成三个不同的apk,还要在应用市场能够上线,麻麻滴这样一听那还不要各个apk包的包名不同以及apk签名文件也不同嘛(签名文件一般也用不同,为防止上线冲突嘛).所以就亲自尝 ...

  7. ASP.NET中彩票项目中的计算复式投注的注数的方法

    从别人做的项目中抽取出的代码:

  8. jfinal想用到中大型项目中的项目经验分享

    jfinal 用在大项目中更加方便实用,节省无数的开发时间,代码量相对 SSH 减少 75% 至 90%,对于项目结构来说,简单提以下几点: 1:先分大模块,大模块内部可以根据划分的model分成子包 ...

  9. Eclipse 中 Java 项目中 .settings 文件夹作用

    今天工作时,因对 .settings 文件夹误操作,耗时 6 个多小时,才了解到原因就出在 .settings 文件夹.经查阅资料,对 .settings 做如下整理: 就如setting这个名字,就 ...

随机推荐

  1. Oracle工作笔记

    重命名表 RENAME PUB_ORDER_DATE_RES TO PUB_ORDER_DATA_RES; 新增字段 ); 删除字段 ALTER TABLE PUB_ORDER_DATA_RES DR ...

  2. Visual Studio 2013百度云下载地址

    Visual Studio 2013百度云下载地址: 链接: https://pan.baidu.com/s/1JkVYLnFo2TWSu4dkDdZP3Q  提取码: 关注公众号[获取   winf ...

  3. 【LOJ】#2064. 「HAOI2016」找相同字符

    题解 做后缀自动机题要一点脑洞,脑洞一开,就过了 我们显然要拿第二个串跑第一个串的后缀自动机 我们可以求出第二个串每个位置匹配到的节点,和匹配的长度L 那么我们统计一个后缀树上的根缀和,表示这样个节点 ...

  4. Spring的配置文件ApplicationContext.xml配置头文件解析

    Spring的配置文件ApplicationContext.xml配置头文件解析 原创 2016年12月16日 14:22:43 标签: spring配置文件 5446 spring中的applica ...

  5. PHP老师没教过你的那些知识点

    另类的写法有惊喜 我们在阅读某些源代码的时候会发现有一种另类的写法,比如 //异常写法 if(false == $result)   //正常写法 if($result == false) 其实这是一 ...

  6. The file will have its original line endings in your working directory.

    在空仓库的情况下,add,出现一下问题 The file will have its original line endings in your working directory. 当报这个警告时是 ...

  7. Java 初相识

    Java是如何出现的呢?这就要回到1991年,那时候随着单片机的发展,出现了很多微型的系统,Sun公司在这个时候就成立的一个项目组,成员就有我们熟知的“Java之父” 詹姆斯·高斯林,起初的目标是为了 ...

  8. 洛谷——P1231 教辅的组成

    P1231 教辅的组成 题目背景 滚粗了的HansBug在收拾旧语文书,然而他发现了什么奇妙的东西. 题目描述 蒟蒻HansBug在一本语文书里面发现了一本答案,然而他却明明记得这书应该还包含一份练习 ...

  9. 1014 Waiting in Line (30)(30 point(s))

    problem Suppose a bank has N windows open for service. There is a yellow line in front of the window ...

  10. Python2字符编码问题汇总

    目录 从字符编码说起 unicode与utf-8 当编解码遇上Python2.x unicode 与 str 区别 __str__ __repr__的区别 unicode str utf-8关系 un ...