java线程池拒绝策略_Java线程池拒绝策略

java线程池拒绝策略_Java线程池拒绝策略Java提供的策略实现CallerRunsPolicypublicstaticclassCallerRunsPolicyimplementsRejectedExecutionHandler{publicCallerRunsPolicy(){}publicvoidrejectedExecution(Runnabler,ThreadPoolExecutore){if(!…

大家好,又见面了,我是你们的朋友全栈君。

Java提供的策略实现

CallerRunsPolicy

public static class CallerRunsPolicy implements RejectedExecutionHandler {

public CallerRunsPolicy() { }

public void rejectedExecution(Runnable r, ThreadPoolExecutor e) {

if (!e.isShutdown()) {

r.run();

}

}

}

CallerRunsPolicy顾名思义就是调用者自己执行,看实现就知道,如果线程池还没有关闭,就让将要被执行的线程自己调用自己的run方法,注意这里不是启动新的线程,只是执行run方法而已。

AbortPolicy

public static class AbortPolicy implements RejectedExecutionHandler {

public AbortPolicy() { }

public void rejectedExecution(Runnable r, ThreadPoolExecutor e) {

throw new RejectedExecutionException(“Task ” + r.toString() +

” rejected from ” +

e.toString());

}

}

AbortPolicy实现非常简单,直接抛出一个RejectedExecutionException异常,ThreadPoolExecutor默认的策略就是这一个。

DiscardPolicy

public static class DiscardPolicy implements RejectedExecutionHandler {

public DiscardPolicy() { }

public void rejectedExecution(Runnable r, ThreadPoolExecutor e) {

}

}

DiscardPolicy策略就是什么也不做,就是把这一个任务(Runnable)丢弃,这是因为ThreadPoolExecutor在执行拒绝策略的时候就不会新建线程来执行Runnable,也不会加入任务队列了,所以在拒绝策略中什么都不做就是简单的丢弃掉了任务(Runnable)

DiscardOldestPolicy

public static class DiscardOldestPolicy implements RejectedExecutionHandler {

public DiscardOldestPolicy() { }

public void rejectedExecution(Runnable r, ThreadPoolExecutor e) {

if (!e.isShutdown()) {

e.getQueue().poll();

e.execute(r);

}

}

}

DiscardOldestPolicy策略不是丢弃提交的任务中最老的,而是丢弃任务队列中队首的任务,然后再次提交新加入的任务。

拒绝策略与模式

从线程池的拒绝策略实现上也体现了面向对象的思想,把拒绝策略抽象在RejectedExecutionHandler接口中,ThreadPoolExecutor关联(更加具体的来说是聚合aggregation)RejectedExecutionHandler接口。

好歹我们也是学过设计模式的,在设计模式中这种模式叫做策略模式(Strategy),虽然实现类的后缀都是Policy但事实上这确实是一个策略模式的应用。对于不同拒绝算法的封装。可以看一下下面的类图。

4a12ec25787afe613c0bb864f65a18fd.png

JDK的源码中有大量的设计模式的使用,有的是非常直接的使用,甚至是为设计模式提供了接口,比如观察着模式。更多的是一些变形的使用,JDK对于设计模式的使用已经到了出神入化,大道无形的地步了。

自定义拒绝策略

既然设计上使用的是策略模式,那么对于我们来说扩展就是相当容易的事情了,简单来一个例子测试一下扩展线程池的策略模式。

import java.math.BigDecimal;

import java.util.concurrent.Executors;

import java.util.concurrent.LinkedBlockingQueue;

import java.util.concurrent.RejectedExecutionHandler;

import java.util.concurrent.ThreadPoolExecutor;

import java.util.concurrent.TimeUnit;

import java.util.concurrent.atomic.AtomicInteger;

public class RejectPolicyTest {

private static AtomicInteger discardTaskNum = new AtomicInteger(0);

private static AtomicInteger executeTaskNum = new AtomicInteger(0);

public static void main(String[] args) {

int cpu = Runtime.getRuntime().availableProcessors();

ThreadPoolExecutor pool = new ThreadPoolExecutor(cpu, cpu, 0L, TimeUnit.MILLISECONDS, new LinkedBlockingQueue(cpu),

Executors.defaultThreadFactory(), new RejectedExecutionHandler() {

@Override

public void rejectedExecution(Runnable r, ThreadPoolExecutor executor) {

System.out.println(r + “:is discarded”);

discardTaskNum.incrementAndGet();

}

});

Task task = new Task();

for(int i = 0;i<100;i++){

pool.submit(task);

try {

TimeUnit.MILLISECONDS.sleep(10);

} catch (InterruptedException e) {

e.printStackTrace();

}

}

pool.shutdown();

while(!pool.isTerminated());//等待所有的任务执行完成,包括任务队列中的任务

System.out.println(“discardNum:”+discardTaskNum.get());

System.out.println(“executeNum:”+executeTaskNum.get());

}

static class Task implements Runnable {

@Override

public void run() {

System.out.println(“task:” + Thread.currentThread().getName() + “is executing!”);

BigDecimal num1 = new BigDecimal(1.23456);

BigDecimal num2 = new BigDecimal(1.23456);

for (int i = 0; i < 5000000; i++) {//在我的机子上大概运行0.5s

num1.multiply(num2);

}

executeTaskNum.incrementAndGet();

}

}

}

在上面的实例中我们简单的实现了一个匿名的RejectedExecutionHandler,打印一下那些任务被丢弃了,顺便记录了一下被丢弃任务的数量。在上面的例子中的任务是执行5百万次BigDecimal类型的乘法。

总结

Java提供的4中默认策略分别是CallerRunsPolicy如果线程池没有关闭直接当前线程执行run方法,AbortPolicy抛出异常,DiscardPolicy丢弃任务,DiscardOldestPolicy丢弃任务队列头的任务。

Java线程池拒绝策略使用的是策略模式,抽象在RejectedExecutionHandler,如果需要扩展只需要实现RejectedExecutionHandler接口就可以了。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/147710.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • React Native(四)——顶部以及底部导航栏实现方式

    React Native(四)——顶部以及底部导航栏实现方式

  • 网络舆情分析与研判的指标还应有哪些_舆情监测是什么

    网络舆情分析与研判的指标还应有哪些_舆情监测是什么网络聚集的人气、展开的场景与揭示的真相,不仅会推动新闻事件的发展、形成网络舆论,甚至还会直接影响社会主流舆论、推动社会变革。对舆情监测主体来说,如何加强对网络舆情的实时全面监测,并对其做出及时反馈、防患于未然;如何利用现代信息技术做好网络舆情分析,从而进行有效引导和控制;如何化解网络舆情危机,实现网络舆情的高效管理是一项任重而道远的任务。在网络舆情分析和管理中,舆情监测主体的业务需求是基础和根本,业务需求的满足与否,是评判网络舆情分析系统的核心指标。网络舆情分析系统的评判指标参考项,如下:网络舆情分析

  • 原生小程序使用 flyio,以及拦截器

    原生小程序使用 flyio,以及拦截器原生小程序的request请求都是异步请求,在实际项目中使用起来很麻烦,自己封装的方法又不够用,所以想到了flyio,不仅可以实现Promise,而且还可以对所有的页面请求进行拦截,使用起来非常的方便。第一件事首先也是先引用flyio.js了。下载地址:flyio.js在utils目录新建http.jsvarFly=require("flyio.js")//引入路径根据自…

  • acwing吧_acwing算法基础

    acwing吧_acwing算法基础小 A 和小 B 在玩一个游戏。首先,小 A 写了一个由 0 和 1 组成的序列 S,长度为 N。然后,小 B 向小 A 提出了 M 个问题。在每个问题中,小 B 指定两个数 l 和 r,小 A 回答 S[l∼r] 中有奇数个 1 还是偶数个 1。机智的小 B 发现小 A 有可能在撒谎。例如,小 A 曾经回答过 S[1∼3] 中有奇数个 1,S[4∼6] 中有偶数个 1,现在又回答 S[1∼6] 中有偶数个 1,显然这是自相矛盾的。请你帮助小 B 检查这 M 个答案,并指出在至少多少个回答之后可

  • m.2接口sata和pcie区别_M2固态硬盘安装方法

    m.2接口sata和pcie区别_M2固态硬盘安装方法http://mst.zol.com.cn/615/6150989.html犹记得当年Windows7系统体验指数中,那5.9分磁盘分数,在其余四项的7.9分面前,似乎已经告诉我们机械硬盘注定被时代

  • zabbix短信报警_手机短信监听

    zabbix短信报警_手机短信监听zabbix短信监控

    2022年10月31日

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号