/**
* 首次启动加载数据至缓存
*/
public class ApplicationStartTask { private static Logger logger = LoggerFactory.getLogger(ApplicationStartTask.class);
private AtomicInteger loadNum = new AtomicInteger(0); @Autowired
private FaceProducer cacheSynchronizationProducer;
@Autowired
private FaceConsumer cacheSynchronizationConsumer; @Autowired
private FaceInfoService faceInfoService; /**
* 当提交一个任务到线程池时,线程池会创建一个线程来执行任务,即使其他空闲的基本线程能够执行新任务也会创建线程,
* <br>等到需要执行的任务数大于线程池基本大小时就不再创建
*/
private static final int POOL_CORE_SIZE = Runtime.getRuntime().availableProcessors(); /**
* 线程池允许创建的最大线程数。如果队列满了,并且已创建的线程数小于最大线程数,则线程池会再创建新的线程执行任务
*/
private static final int POOL_MAX_SIZE = Runtime.getRuntime().availableProcessors(); /**
* 线程池的工作线程空闲后,保持存活的时间 <br>
* 单位:分
*/
private static final int POOL_KEEP_ALIVE_TIME = 2; @Autowired
@Qualifier("sqliteManagerImpl")
private SqlieteManager sqlieteManager; @Autowired
private FaceCompareManager faceCompareManager; @Value("${local.sqlite.sqlname}")
private String sqlName; @Bean
public ApplicationStartTask loadCache() {
logger.info("[启动任务]首次启动载入数据至缓存...");
checkSqLiteFile();
List<String> belongIds = sqlieteManager.getAllBelongIds();
if (ArrayUtil.isNotEmpty(belongIds)) { loadFromSqlite(belongIds); initMqServer(); return new ApplicationStartTask();
}
loadFromDB(); initMqServer();
return new ApplicationStartTask(); } /**
* 开启MQ服务
*/
private void initMqServer() { logger.info("开始开启MQ服务...");
cacheSynchronizationProducer.producerStart();
cacheSynchronizationConsumer.getMessage();
} /**
* 直接从MYSQL载入数据
*/
private void loadFromDB() {
logger.info("当前CPU内核线程数:{}", POOL_CORE_SIZE); long beginTimsamp = System.currentTimeMillis(); List<String> allBelongId = faceCompareManager.getAllBelongId();
int size = 0;
for (String belongId : allBelongId) {
FaceInfoSyncDataQuery query = new FaceInfoSyncDataQuery(belongId);
query.setPageSize(1000);
query.setDeleteStatus(0);
Result<Integer> result = faceInfoService.getTotalPage(query);
if (!result.isSuccess() || result.getModel() == 0) {
continue;
} int totalPage = result.getModel(); size += totalPage; }
logger.info("总任务数:{}", size);
ThreadPoolExecutor poolExecutor = new ThreadPoolExecutor(POOL_CORE_SIZE, POOL_MAX_SIZE, POOL_KEEP_ALIVE_TIME,
TimeUnit.MINUTES, new LinkedBlockingQueue<Runnable>(size));
CountDownLatch latch = new CountDownLatch(size); for (String belongId : allBelongId) { FaceInfoSyncDataQuery query = new FaceInfoSyncDataQuery();
query.setBelongId(belongId);
query.setDeleteStatus(0);
query.setPageSize(1000);
Result<FaceInfoSyncDataVO> result = faceInfoService.getFaces(query);
if (!result.isSuccess() || result.getModel() == null || result.getModel().getTotalPages() == 0) {
continue;
}
FaceInfoSyncDataVO faceInfoSyncDataVO = result.getModel(); int totalPage = faceInfoSyncDataVO.getTotalPages();
//第一页 poolExecutor.execute(new CacheLoadTask1(1, belongId, faceInfoSyncDataVO.getFaces(), latch));
if (totalPage == 1) {
continue;
}
for (int i = 2; i <= totalPage; i++) {
query = new FaceInfoSyncDataQuery();
query.setBelongId(belongId);
query.setPageSize(1000);
query.setPageNo(i);
query.setDeleteStatus(0);
result = faceInfoService.getFaces(query);
if (!result.isSuccess() || result.getModel() == null || ArrayUtil.isEmpty(result.getModel().getFaces())) {
continue;
}
faceInfoSyncDataVO = result.getModel(); poolExecutor.execute(new CacheLoadTask1(i, belongId, faceInfoSyncDataVO.getFaces(), latch)); }
}
try { latch.await();// 同步阻塞,直到所有线程工作完成
// long endTime = System.currentTimeMillis(); // 获取结束时间 poolExecutor.shutdown();// 不允许提交新任务,等待当前任务及队列中的任务全部执行完毕后退出
// // 支持等待以前提交的任务停止执行
// // 所有任务关闭请求或线程中断或超时,阻塞取消
// poolExecutor.awaitTermination(20, TimeUnit.MINUTES);
// poolExecutor.shutdownNow();//通过Thread.interrupt试图停止所有正在执行的线程,并不再处理还在队列中等待的任务...
} catch (Exception e) {
logger.error("[启动任务]线程池异常打印:{}", e);
poolExecutor.shutdownNow(); }
// logger.info("[启动任务]从DB中载入数据至缓存完毕!共载入{}条人脸数据", loadNum.get()); logger.info("[启动任务]完毕!全文索引{}条数据,共耗时:{}s", loadNum.get(), (System.currentTimeMillis() - beginTimsamp) / 1000); } class CacheLoadTask1 implements Runnable { private CountDownLatch latch;
List<FaceInfoDO> faceInfoDOs = null;
int putCacheNum = 0;
String belongId = null;
int pageNo = 0; public CacheLoadTask1(int pageNo, String belongId, List<FaceInfoDO> faceInfoDOs, CountDownLatch latch) {
this.latch = latch;
this.faceInfoDOs = faceInfoDOs;
this.belongId = belongId;
this.pageNo = pageNo;
// this.index = index;
} @Override
public void run() {
Long beginTime = System.currentTimeMillis();
try {
for (FaceInfoDO faceInfoDO : faceInfoDOs) {
boolean isPutCache = FaceInfoCacheManage.put(faceInfoDO);
if (isPutCache) {
putCacheNum++;
loadNum.incrementAndGet(); }
}
logger.info("[启动任务]线程{}处理完belongId-{}第{}页数据,共载入{}条人脸数据,耗时:{}ms", Thread.currentThread().getName(),
belongId, pageNo, putCacheNum, System.currentTimeMillis() - beginTime); } catch (Exception e) {
logger.error("[启动任务]载入sqlite任务[CacheLoadTask]出错:{}", e);
} finally {
// TODO: handle finally clause
latch.countDown();
}
} } /**
* 从sqlite数据库中加载数据
*/
private void loadFromSqlite(List<String> belongIds) { logger.info("当前CPU内核线程数:{}", POOL_CORE_SIZE);
ThreadPoolExecutor poolExecutor = new ThreadPoolExecutor(POOL_CORE_SIZE, POOL_MAX_SIZE, POOL_KEEP_ALIVE_TIME,
TimeUnit.MINUTES, new LinkedBlockingQueue<Runnable>(belongIds.size()));
CountDownLatch latch = new CountDownLatch(belongIds.size());
for (String belongId : belongIds) {
List<FaceInfoDO> faceInfoDOs = sqlieteManager.selectAllInfoByBelongId(sqlName, belongId); poolExecutor.execute(new CacheLoadTask(belongId, faceInfoDOs, latch)); } try { latch.await();// 同步阻塞,直到所有线程工作完成
// long endTime = System.currentTimeMillis(); // 获取结束时间
logger.info("[启动任务]从sqlite中载入数据至缓存完毕!共载入{}条人脸数据", loadNum.get()); poolExecutor.shutdown();// 不允许提交新任务,等待当前任务及队列中的任务全部执行完毕后退出
// // 支持等待以前提交的任务停止执行
// // 所有任务关闭请求或线程中断或超时,阻塞取消
// poolExecutor.awaitTermination(20, TimeUnit.MINUTES);
// poolExecutor.shutdownNow();//通过Thread.interrupt试图停止所有正在执行的线程,并不再处理还在队列中等待的任务...
} catch (Exception e) {
logger.error("[启动任务]线程池异常打印:{}", e);
poolExecutor.shutdownNow(); } } private void checkSqLiteFile() {
File sqliteFile = new File(sqlName + ".sqlite3");
if (!sqliteFile.exists()) {
logger.info("[启动任务]sqlite数据库未创建,开始创建...");
// 第一次启动创建本地表
try {
sqlieteManager.createTable(sqlName);
} catch (Exception e) {
logger.error(e.getMessage());
} }
} }


多线程并行_countDown的更多相关文章

  1. Java多线程--并行模式与算法

    Java多线程--并行模式与算法 单例模式 虽然单例模式和并行没有直接关系,但是我们经常会在多线程中使用到单例.单例的好处有: 对于频繁使用的对象可以省去new操作花费的时间: new操作的减少,随之 ...

  2. java8新特性(六):Stream多线程并行数据处理

    转:http://blog.csdn.net/sunjin9418/article/details/53143588 将一个顺序执行的流转变成一个并发的流只要调用 parallel()方法 publi ...

  3. MySQL 5.7主从复制从零开始设置及全面详解——实现多线程并行同步,解决主从复制延迟问题!

    MySQL 5.7主从复制从零开始设置及全面详解——实现多线程并行同步,解决主从复制延迟问题!2017年06月15日 19:59:44 蓝色-鸢尾 阅读数:2062版权声明:本文为博主原创文章,如需转 ...

  4. Python基础补充(二) 多核CPU上python多线程并行的一个假象【转】

    在python上开启多个线程,由于GIL的存在,每个单独线程都会在竞争到GIL后才运行,这样就干预OS内部的进程(线程)调度,结果在多核CPU上: python的多线程实际是串行执行的,并不会同一时间 ...

  5. C#中的多线程 - 并行编程 z

    原文:http://www.albahari.com/threading/part5.aspx 专题:C#中的多线程 1并行编程Permalink 在这一部分,我们讨论 Framework 4.0 加 ...

  6. Oracle多线程并行使用、关联与指定索引执行

    nologging AS SELECT /*+parallel(4) leading(s a) use_hash(A) index(s IDX_CS_SERVICE_RECORD_MD2_04) */ ...

  7. Optaplanner终于支持多线程并行运行 - Multithreaded incremental solving

    Optaplanner 7.9.0.Final之前,启动引擎开始对一个Problem进行规划的时候,只能是单线程进行的.也就是说,当引擎对每一个possible solution进行分数计算的过程中, ...

  8. 大数据应用之HBase数据插入性能优化之多线程并行插入测试案例

    一.引言: 上篇文章提起关于HBase插入性能优化设计到的五个参数,从参数配置的角度给大家提供了一个性能测试环境的实验代码.根据网友的反馈,基于单线程的模式实现的数据插入毕竟有限.通过个人实测,在我的 ...

  9. .Net多线程 并行编程(三)---并行集合

    为了让共享的数组,集合能够被多线程更新,我们现在(.net4.0之后)可以使用并发集合来实现这个功能. 而System.Collections和System.Collections.Generic命名 ...

随机推荐

  1. 20190626_二次开发BarTender打印机_C#代码_一边读取TID_一边打印_打印机POSTEK

    demo代码如下: private void btnPrint_Click(object sender, EventArgs e) { if (this.btnPrint.Text == " ...

  2. moviepy音视频剪辑:与大小相关的视频变换函数crop、even_size、margin、resize介绍

    ☞ ░ 前往老猿Python博文目录 ░ 一.引言 在<moviepy音视频剪辑:moviepy中的剪辑基类Clip详解>介绍了剪辑基类的fl.fl_time.fx方法,在<movi ...

  3. 第10.10节 Python使用__init__.py自动加载包下内容

    在前面章节老猿介绍了包下模块及子包的加载的各种方式,并说明包的加载首先是自动加载包下的__init__.py文件.在<第10.6节 Python包的概念>中介绍了__init__.py文件 ...

  4. PyQt(Python+Qt)学习随笔:调用disconnect进行信号连接断开时的信号签名与断开参数的匹配要求

    老猿Python博文目录 专栏:使用PyQt开发图形界面Python应用 老猿Python博客地址 在使用信号调用disconnect()方法断开信号和槽的连接时,信号可以带签名也可不带签名,参数可以 ...

  5. java中的反射(二)

    java中的反射(一):https://www.cnblogs.com/KeleLLXin/p/14060555.html 目录 一.反射 1.class类 2.访问字段 3.调用方法 4.调用构造方 ...

  6. 补:冲刺Day2

    每天举行站立式会议照片: 昨天已完成的工作: 各个成员在 Alpha 阶段认领的任务. 今天各个成员的任务安排. 冲刺Day1博客. 今天计划完成的工作: 成员 任务 高嘉淳 完成登陆.注册 覃泽泰 ...

  7. 动态规划之经典数学期望和概率DP

    起因:在一场训练赛上.有这么一题没做出来. 题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6829 题目大意:有三个人,他们分别有\(X,Y,Z\)块钱 ...

  8. linux 设置别名

    [root@oldboyedu45-xy data]# alias(查找别名) alias cp='cp -i' alias l.='ls -d .* --color=auto' alias ll=' ...

  9. Linux查看、开启、关闭防火墙操作

    一.防火墙区别 CentOS6自带的防火墙是iptables,CentOS7自带的防火墙是firewall. iptables:用于过滤数据包,属于网络层防火墙. firewall:底层还是使用 ip ...

  10. 【Go语言绘图】图片添加文字(一)

    前一篇讲解了利用gg包来进行图片旋转的操作,这一篇我们来看看怎么在图片上添加文字. 绘制纯色背景 首先,我们先绘制一个纯白色的背景,作为添加文字的背景板. package main import &q ...