1.前言

第二篇设计模式的文章我们谈谈Visitor模式。

当然,不是简单的列个的demo,我们以电商网站中的购物车功能为背景,使用线程池实现并发的Visitor模式,并聊聊其中的几个关键点。

  • 一,基于线程池的实现并发Visitor模式。
  • 二,讨论下并发场景下的一些细节处理。
  • 三,用模拟数据测试并做补充说明。

2.背景

当从网站的某个页面进入购物车时,服务端需要做各种数据处理,比如刷新商品价格,计算促销价、校验库存等等。这些操作会随着业务的增加不断扩展,那么Visitor模式就适合这种场景,这也是它的优点之一,易于新增操作。

3.实现

上图是书上的UML原图,原生实现这里就不重复了,我们直接进入正题。

3.1 并发的visit操作

先来看关键的visit方法

public void visit(Cart cart) {
// 当商品数量不超过两个时,不做并发处理,直接主线程执行,避免浪费线程资源
if (cart.getItems().size() <= 2) {
for (AbstractItem item : cart.getItems()) {
item.accept(this);
}
} else {
//CountDownLatch 用来保证并发执行之后的同步,计数器为商品数量,只有所有visit操作完成后才继续主流程
CountDownLatch countDownLatch = new CountDownLatch(cart.getItems().size());
// 提交task到线程池
cart.getItems().forEach(item -> {
threadPoolExecutor.submit(new SubVisitor(item, countDownLatch));
});
try {
// 等待所有子线程执行完毕
countDownLatch.await();
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

SubVisitor很简单,实现Runnalbe接口,负责执行具体的visit逻辑,并在执行完成之后同步操作。

private class SubVisitor implements Runnable {
CountDownLatch countDownLatch;
AbstractItem abstractItem;
public SubVisitor(AbstractItem abstractItem, CountDownLatch countDownLatch) {
this.abstractItem = abstractItem;
this.countDownLatch = countDownLatch;
}
@Override
public void run() {
abstractItem.accept(AbstractCartVisitor.this);
// 执行完毕之后同步countDownLatch,即计算器减一
countDownLatch.countDown();
}
}

这里只需注意两点,

  • 在商品数量不超过两个时不使用线程池,以避免浪费资源。
  • 通过并发包中CountDownLatch来同步所有操作。

3.2 线程池配置

另外一个核心的配置是线程池,代码如下。

    int availableProcessors = Runtime.getRuntime().availableProcessors();
threadPoolExecutor = new ThreadPoolExecutor(availableProcessors, availableProcessors,
10, TimeUnit.MINUTES, new SynchronousQueue<Runnable>(), new ThreadPoolExecutor.CallerRunsPolicy());

corePoolSize和maximumPoolSize配置为当前cpu核数,之所以一致是因为这样可以让线程池一直处于最大运行状态,我们追求的是最大效率,不需要销毁一部分来达到减少内存占用。

接下来的参数workQueue是线程池的缓冲队列,超出容量的任务会被放入此队列,这里使用SynchronousQueue,它是一个零容量队列,意味着线程池没有缓冲区。

RejectedExecutionHandler是则是拒绝策略,当线程池和缓冲区都满了后将会执行拒绝策略。这里我们使用CallerRunsPolicy,翻译过来就是主线程直接执行,我们可以看下它的实现。

public void rejectedExecution(Runnable r, ThreadPoolExecutor e) {
if (!e.isShutdown()) {
r.run();
}
}

这样的RejectedExecutionHandler和workQueue的配置可以达到我们想要的一个效果:

如果线程池未满,则将任务提交给线程池执行;

线程池满了后,不缓冲,由主线程直接执行任务,不阻塞请求。

3.3 测试

我们模拟一个刷新价格的测试场景来看一下效果。

为了简单起见,我们将corePoolSize设为4。

new ThreadPoolExecutor(4, 4 ...

并假设价格查询接口耗时20ms:

public double queryPriceByItemId(int id) {
//模拟调用时间20毫秒
try {
Thread.sleep(20);
} catch (InterruptedException e) {
}
return 1;
}

以下为1~6个商品时执行结果

数量 1 金额 1.0 耗时  23
数量 2 金额 2.0 耗时 40
数量 3 金额 3.0 耗时 20
数量 4 金额 4.0 耗时 20
数量 5 金额 5.0 耗时 21
数量 6 金额 6.0 耗时 40

2个及以下商品时耗时为线性增长,这是由于我们对这种场景没做并发处理。

4个商品时耗时为20ms,符合我们的预期。

为什么5个的时候也是20ms?

其原因在于我们的执行策略是SynchronousQueue+CallerRunsPolicy,第5个商品被主线程直接处理了,因此相当于我们实际的执行线程是5个,也达到了预期效果。

#
以上为并发的Visitor模式的全部内容,完成demo见Github:

https://github.com/wchukai/basic-practice

作者:初开

原文链接:https://wchukai.com/article/design-patterns-concurrent-visitor-patterns-based-on-thread-pools

本文由MetaCLBlog于2017-10-24 09:00:51自动同步至cnblogs

本文基于 知识共享-署名-非商业性使用-禁止演绎 4.0 国际许可协议发布,转载必须保留署名及链接。

设计模式:基于线程池的并发Visitor模式的更多相关文章

  1. python 并发编程 基于线程池实现并发的套接字通信

    不应该让服务端随着 并发的客户端数量增多,而无数起线程,应该用线程池,限制线程数量,控制最大并发数 io密集型程序,最大并发数是2 客户端 from socket import * client = ...

  2. 基于线程池的多并发Socket程序的实现

    Socket“服务器-客户端”模型的多线程并发实现效果的大体思路是:首先,在Server端建立“链接循环”,每一个链接都开启一个“线程”,使得每一个Client端都能通过已经建立好的线程来同时与Ser ...

  3. Java线程池(Callable+Future模式)

    转: Java线程池(Callable+Future模式) Java线程池(Callable+Future模式) Java通过Executors提供四种线程池 1)newCachedThreadPoo ...

  4. 基于线程池的多线程售票demo2.0(原创)

    继上回基于线程池的多线程售票demo,具体链接: http://www.cnblogs.com/xifenglou/p/8807323.html以上算是单机版的实现,特别使用了redis 实现分布式锁 ...

  5. 基于线程池的多线程售票demo(原创)

    废话不多说,直接就开撸import org.springframework.util.StopWatch;import java.util.concurrent.*;/** * 基于线程池实现的多线程 ...

  6. requests模块session处理cookie 与基于线程池的数据爬取

    引入 有些时候,我们在使用爬虫程序去爬取一些用户相关信息的数据(爬取张三“人人网”个人主页数据)时,如果使用之前requests模块常规操作时,往往达不到我们想要的目的,例如: #!/usr/bin/ ...

  7. 【Java TCP/IP Socket】基于线程池的TCP服务器(含代码)

    了解线程池 在http://blog.csdn.net/ns_code/article/details/14105457(读书笔记一:TCP Socket)这篇博文中,服务器端采用的实现方式是:一个客 ...

  8. Python网络爬虫之cookie处理、验证码识别、代理ip、基于线程池的数据爬去

    本文概要 session处理cookie proxies参数设置请求代理ip 基于线程池的数据爬取 引入 有些时候,我们在使用爬虫程序去爬取一些用户相关信息的数据(爬取张三“人人网”个人主页数据)时, ...

  9. Java多线程设计模式(4)线程池模式

    前序: Thread-Per-Message Pattern,是一种对于每个命令或请求,都分配一个线程,由这个线程执行工作.它将“委托消息的一端”和“执行消息的一端”用两个不同的线程来实现.该线程模式 ...

随机推荐

  1. java课程设计-猜数游戏(201521123029 郑佳明)

    1.团队课程设计博客链接 http://www.cnblogs.com/m1ng123/p/7056740.html 2.个人负责模板或任务说明 猜数运行3个主界面即相关功能 玩家信息存储的play类 ...

  2. Java:Object类的equals()方法 如何编写一个完美的equals()方法

    一  代码实例: package freewill.objectequals;  /** * @author freewill * @see Core Java page161 * @desc get ...

  3. 压缩空格的函数以及BCD码与ASCII相互转换函数

    /**函数名:zip_space功能 :压缩空格的函数参数 : s 源字符串返回值: 0 成功**/ int zip_space(char *s){ int i,j; int len; ) { ; } ...

  4. Oracle_Sequence如何初始化开始值

    Sequence的start with 值如何确定才能保证生成的主键不会冲突??? 我的项目中最开始数据库表主键的生成策略是 increment,但由于后来采用了集群部署的方式,出现了主键冲突的问题. ...

  5. 自己把jar包添加到maven仓库中

    定制库到Maven本地资源库 这里有2个案例,需要手动发出Maven命令包括一个 jar 到 Maven 的本地资源库. 要使用的 jar 不存在于 Maven 的中心储存库中. 您创建了一个自定义的 ...

  6. 框架应用:Spring framework (一) - IoC技术

    IoC概念以及目标 IoC就是让原本你自己管理的对象交由容器来进行管理,其主要的目的是松耦合. IoC发展史 既然IoC的目标是为了松耦合,那它怎么做到的? 最后目标:降低对象之间的耦合度,IoC技术 ...

  7. 深入理解计算机系统(2.7)------二进制小数和IEEE浮点标准

    整数的表示和运算我们已经讲完了,在实际应用中,整数能够解决我们大部分问题.但是某些需要精确表示的数,比如某件商品的价格,某两地之间的距离等等,我们如果用整数表示将会有很大的出入,这时候浮点数就产生了. ...

  8. AngularJS [ 快速入门教程 ]

      前  序 S    N AngularJS是什么? 我想既然大家查找AngularJS就证明大家多多少少对AngularJS都会有了解. AngularJS就是,使用JavaScript编写的客户 ...

  9. Java 自增(++) 和 C语言中自增的区别

    在Java.c语言等高级语言中自增和自减的作用基本一致,都是变量自身加一或减一.下面我只对自增进行说明,自减是类似的. 自增运算符(++),有两种书写形式,一个是在变量前: ++ num; 另一种在变 ...

  10. Redis 的安装与使用

    环境:CentOS 6.6 Redis 版本:redis-3.0 (考虑到 Redis3.0 在集群和性能提升方面的特性,rc 版为正式版的候选版,而且很快就出正式版) 安装目录:/usr/local ...