当前位置: 首页 > news >正文

昆明网站建设价格低南宁网站建设企业网站

昆明网站建设价格低,南宁网站建设企业网站,传媒公司合同模板,企业宣传片拍摄思路一 重试框架之Spring-Retry 1.Spring-Retry的普通使用方式 2.Spring-Retry的注解使用方式 二 重试框架之Guava-Retry 总结 图片 一 重试框架之Spring-Retry Spring Retry 为 Spring 应用程序提供了声明性重试支持。它用于Spring批处理、Spring集成、Apache Hadoop(等等)。… 一 重试框架之Spring-Retry 1.Spring-Retry的普通使用方式 2.Spring-Retry的注解使用方式 二 重试框架之Guava-Retry 总结 图片 一 重试框架之Spring-Retry Spring Retry 为 Spring 应用程序提供了声明性重试支持。它用于Spring批处理、Spring集成、Apache Hadoop(等等)。它主要是针对可能抛出异常的一些调用操作进行有策略的重试 1. Spring-Retry的普通使用方式 1.准备工作 我们只需要加上依赖: dependencygroupIdorg.springframework.retry/groupIdartifactIdspring-retry/artifactIdversion1.2.2.RELEASE/version/dependency准备一个任务方法我这里是采用一个随机整数根据不同的条件返回不同的值或者抛出异常 package com.zgd.demo.thread.retry;import lombok.extern.slf4j.Slf4j; import org.apache.commons.lang3.RandomUtils; import org.springframework.remoting.RemoteAccessException;/*** Author: zgd* Description:*/ Slf4j public class RetryDemoTask {/*** 重试方法* return*/public static boolean retryTask(String param)  {log.info(收到请求参数:{},param);int i  RandomUtils.nextInt(0,11);log.info(随机生成的数:{},i);if (i  0) {log.info(为0,抛出参数异常.);throw new IllegalArgumentException(参数异常);}else if (i   1){log.info(为1,返回true.);return true;}else if (i  2){log.info(为2,返回false.);return false;}else{//为其他log.info(大于2,抛出自定义异常.);throw new RemoteAccessException(大于2,抛出远程访问异常);}}}2.使用SpringRetryTemplate 这里可以写我们的代码了 package com.zgd.demo.thread.retry.spring;import com.zgd.demo.thread.retry.RetryDemoTask; import lombok.extern.slf4j.Slf4j; import org.junit.Test; import org.springframework.remoting.RemoteAccessException; import org.springframework.retry.backoff.FixedBackOffPolicy; import org.springframework.retry.policy.SimpleRetryPolicy; import org.springframework.retry.support.RetryTemplate;import java.util.HashMap; import java.util.Map;/*** Author: zgd* Description: spring-retry 重试框架*/ Slf4j public class SpringRetryTemplateTest {/*** 重试间隔时间ms,默认1000ms* */private long fixedPeriodTime  1000L;/*** 最大重试次数,默认为3*/private int maxRetryTimes  3;/*** 表示哪些异常需要重试,key表示异常的字节码,value为true表示需要重试*/private MapClass? extends Throwable, Boolean exceptionMap  new HashMap();Testpublic void test() {exceptionMap.put(RemoteAccessException.class,true);// 构建重试模板实例RetryTemplate retryTemplate  new RetryTemplate();// 设置重试回退操作策略主要设置重试间隔时间FixedBackOffPolicy backOffPolicy  new FixedBackOffPolicy();backOffPolicy.setBackOffPeriod(fixedPeriodTime);// 设置重试策略主要设置重试次数SimpleRetryPolicy retryPolicy  new SimpleRetryPolicy(maxRetryTimes, exceptionMap);retryTemplate.setRetryPolicy(retryPolicy);retryTemplate.setBackOffPolicy(backOffPolicy);Boolean execute  retryTemplate.execute(//RetryCallbackretryContext - {boolean b  RetryDemoTask.retryTask(abc);log.info(调用的结果:{}, b);return b;},retryContext - {//RecoveryCallbacklog.info(已达到最大重试次数或抛出了不重试的异常~~~);return false;});log.info(执行结果:{},execute);}}简单剖析下案例代码RetryTemplate 承担了重试执行者的角色它可以设置SimpleRetryPolicy(重试策略设置重试上限重试的根源实体)FixedBackOffPolicy固定的回退策略设置执行重试回退的时间间隔。 RetryTemplate通过execute提交执行操作需要准备RetryCallback 和RecoveryCallback 两个类实例前者对应的就是重试回调逻辑实例包装正常的功能操作RecoveryCallback实现的是整个执行操作结束的恢复操作实例. 只有在调用的时候抛出了异常并且异常是在exceptionMap中配置的异常才会执行重试操作否则就调用到excute方法的第二个执行方法RecoveryCallback中 当然,重试策略还有很多种,回退策略也是: 重试策略 NeverRetryPolicy 只允许调用RetryCallback一次不允许重试 AlwaysRetryPolicy 允许无限重试直到成功此方式逻辑不当会导致死循环 SimpleRetryPolicy 固定次数重试策略默认重试最大次数为3次RetryTemplate默认使用的策略 TimeoutRetryPolicy 超时时间重试策略默认超时时间为1秒在指定的超时时间内允许重试 ExceptionClassifierRetryPolicy 设置不同异常的重试策略类似组合重试策略区别在于这里只区分不同异常的重试 CircuitBreakerRetryPolicy 有熔断功能的重试策略需设置3个参数openTimeout、resetTimeout和delegate CompositeRetryPolicy 组合重试策略有两种组合方式乐观组合重试策略是指只要有一个策略允许即可以重试悲观组合重试策略是指只要有一个策略不允许即可以重试但不管哪种组合方式组合中的每一个策略都会执行 重试回退策略 重试回退策略指的是每次重试是立即重试还是等待一段时间后重试。 默认情况下是立即重试如果需要配置等待一段时间后重试则需要指定回退策略BackoffRetryPolicy。 NoBackOffPolicy 无退避算法策略每次重试时立即重试 FixedBackOffPolicy 固定时间的退避策略需设置参数sleeper和backOffPeriodsleeper指定等待策略默认是Thread.sleep即线程休眠backOffPeriod指定休眠时间默认1秒 UniformRandomBackOffPolicy 随机时间退避策略需设置sleeper、minBackOffPeriod和maxBackOffPeriod该策略在minBackOffPeriod,maxBackOffPeriod之间取一个随机休眠时间minBackOffPeriod默认500毫秒maxBackOffPeriod默认1500毫秒 ExponentialBackOffPolicy 指数退避策略需设置参数sleeper、initialInterval、maxInterval和multiplierinitialInterval指定初始休眠时间默认100毫秒maxInterval指定最大休眠时间默认30秒multiplier指定乘数即下一次休眠时间为当前休眠时间*multiplier ExponentialRandomBackOffPolicy 随机指数退避策略引入随机乘数可以实现随机乘数回退 我们可以根据自己的应用场景和需求使用不同的策略不过一般使用默认的就足够了。 上面的代码的话,我简单的设置了重试间隔为1秒重试的异常是RemoteAccessException下面就是测试代码的情况: 重试第二次成功的情况: 图片 重试一次以后遇到了没有指出需要重试的异常直接结束重试调用retryContext 图片 重试了三次后达到了最大重试次数调用retryContext 图片 2. Spring-Retry的注解使用方式 既然是Spring家族的东西那么自然就支持和Spring-Boot整合 1.准备工作 依赖: dependencygroupIdorg.springframework.retry/groupIdartifactIdspring-retry/artifactIdversion1.2.2.RELEASE/version/dependencydependencygroupIdorg.aspectj/groupIdartifactIdaspectjweaver/artifactIdversion1.9.1/version/dependency2.代码 在application启动类上加上EnableRetry的注解 EnableRetry public class Application {... }为了方便测试我这里写了一个SpringBootTest的测试基类需要使用SpringBootTest的只要继承这个类就好了 package com.zgd.demo.thread.test;/*** Author: zgd* Description:*/import com.zgd.demo.thread.Application; import lombok.extern.slf4j.Slf4j; import org.junit.After; import org.junit.Before; import org.junit.runner.RunWith; import org.springframework.boot.test.context.SpringBootTest; import org.springframework.test.context.junit4.SpringRunner;/*** Author: zgd* Date: 18/09/29 20:33* Description:*/ RunWith(SpringRunner.class) SpringBootTest(classes  Application.class) Slf4j public class MyBaseTest {Beforepublic void init() {log.info(----------------测试开始---------------);}Afterpublic void after() {log.info(----------------测试结束---------------);}}我们只要在需要重试的方法上加Retryable在重试失败的回调方法上加Recover下面是这些注解的属性 图片 建一个service类 package com.zgd.demo.thread.retry.spring;import com.zgd.demo.thread.retry.RetryDemoTask; import com.zgd.demo.thread.test.MyBaseTest; import lombok.extern.slf4j.Slf4j; import org.springframework.remoting.RemoteAccessException; import org.springframework.retry.ExhaustedRetryException; import org.springframework.retry.annotation.Backoff; import org.springframework.retry.annotation.Recover; import org.springframework.retry.annotation.Retryable; import org.springframework.stereotype.Component;/*** Author: zgd* Description:*/ Service Slf4j public class SpringRetryDemo   {/*** 重试所调用方法* param param* return*/Retryable(value  {RemoteAccessException.class},maxAttempts  3,backoff  Backoff(delay  2000L,multiplier  2))public boolean call(String param){return RetryDemoTask.retryTask(param);}/*** 达到最大重试次数,或抛出了一个没有指定进行重试的异常* recover 机制* param e 异常*/Recoverpublic boolean recover(Exception e,String param) {log.error(达到最大重试次数,或抛出了一个没有指定进行重试的异常:,e);return false;}}然后我们调用这个service里面的call方法 package com.zgd.demo.thread.retry.spring;import com.zgd.demo.thread.test.MyBaseTest; import lombok.extern.slf4j.Slf4j; import org.junit.Test; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Component;/*** Author: zgd* Description:*/ Component Slf4j public class SpringRetryDemoTest extends MyBaseTest {Autowiredprivate SpringRetryDemo springRetryDemo;Testpublic void retry(){boolean abc  springRetryDemo.call(abc);log.info(--结果是:{}--,abc);}}这里我依然是RemoteAccessException的异常才重试Backoff(delay 2000L,multiplier 2))表示第一次间隔2秒以后都是次数的2倍,也就是第二次4秒第三次6秒. 来测试一下: 遇到了没有指定重试的异常,这里指定重试的异常是 Retryable(value {RemoteAccessException.class}...,所以抛出参数异常IllegalArgumentException的时候直接回调Recover的方法 图片 重试达到最大重试次数时调用Recover的方法 图片 重试到最后一次没有报错返回false 图片 二 重试框架之Guava-Retry Guava retryer工具与spring-retry类似都是通过定义重试者角色来包装正常逻辑重试但是Guava retryer有更优的策略定义在支持重试次数和重试频度控制基础上能够兼容支持多个异常或者自定义实体对象的重试源定义让重试功能有更多的灵活性。 Guava Retryer也是线程安全的入口调用逻辑采用的是Java.util.concurrent.Callable的call方法示例代码如下 pom.xml加入依赖 !-- https://mvnrepository.com/artifact/com.github.rholder/guava-retrying --dependencygroupIdcom.github.rholder/groupIdartifactIdguava-retrying/artifactIdversion2.0.0/version/dependency更改一下测试的任务方法 package com.zgd.demo.thread.retry;import lombok.extern.slf4j.Slf4j; import org.apache.commons.lang3.RandomUtils; import org.springframework.remoting.RemoteAccessException;/*** Author: zgd* Description:*/ Slf4j public class RetryDemoTask {/*** 重试方法* return*/public static boolean retryTask(String param)  {log.info(收到请求参数:{},param);int i  RandomUtils.nextInt(0,11);log.info(随机生成的数:{},i);if (i  2) {log.info(为0,抛出参数异常.);throw new IllegalArgumentException(参数异常);}else if (i   5){log.info(为1,返回true.);return true;}else if (i  7){log.info(为2,返回false.);return false;}else{//为其他log.info(大于2,抛出自定义异常.);throw new RemoteAccessException(大于2,抛出自定义异常);}}}Guava 这里设定跟Spring-Retry不一样我们可以根据返回的结果来判断是否重试比如返回false我们就重试 package com.zgd.demo.thread.retry.guava;import com.github.rholder.retry.*; import com.zgd.demo.thread.retry.RetryDemoTask; import org.junit.Test; import org.springframework.remoting.RemoteAccessException;import java.util.concurrent.ExecutionException; import java.util.concurrent.TimeUnit; import java.util.function.Predicate;/*** Author: zgd* Description:*/ public class GuavaRetryTest {Testpublic void fun01(){// RetryerBuilder 构建重试实例 retryer,可以设置重试源且可以支持多个重试源可以配置重试次数或重试超时时间以及可以配置等待时间间隔RetryerBoolean retryer  RetryerBuilder.Boolean newBuilder().retryIfExceptionOfType(RemoteAccessException.class)//设置异常重试源.retryIfResult(res- resfalse)  //设置根据结果重试.withWaitStrategy(WaitStrategies.fixedWait(3, TimeUnit.SECONDS)) //设置等待间隔时间.withStopStrategy(StopStrategies.stopAfterAttempt(3)) //设置最大重试次数.build();try {retryer.call(() - RetryDemoTask.retryTask(abc));} catch (Exception e) {e.printStackTrace();}}}运行测试一下 遇到了我们指定的需要重试的异常进行重试间隔是3秒 图片 重试次数超过了最大重试次数 图片 返回为true直接结束重试 图片 遇到了没有指定重试的异常结束重试 图片 返回false重试 图片 我们可以更灵活的配置重试策略比如: retryIfException retryIfException抛出 runtime 异常、checked 异常时都会重试但是抛出 error 不会重试。 retryIfRuntimeException retryIfRuntimeException 只会在抛 runtime 异常的时候才重试checked 异常和error 都不重试。 retryIfExceptionOfType retryIfExceptionOfType 允许我们只在发生特定异常的时候才重试比如NullPointerException 和 IllegalStateException 都属于 runtime 异常也包括自定义的error。 如: retryIfExceptionOfType(NullPointerException.class)// 只在抛出空指针异常重试retryIfResult: retryIfResult 可以指定你的 Callable 方法在返回值的时候进行重试如 // 返回false重试   .retryIfResult(Predicates.equalTo(false))   //以_error结尾才重试   .retryIfResult(Predicates.containsPattern(_error$))//返回为空时重试 .retryIfResult(res- resnull)RetryListener 当发生重试之后假如我们需要做一些额外的处理动作比如log一下异常那么可以使用RetryListener。每次重试之后guava-retrying 会自动回调我们注册的监听。可以注册多个RetryListener会按照注册顺序依次调用。 .withRetryListener(new RetryListener {      Override    public T void onRetry(AttemptT attempt) {  logger.error(第【{}】次调用失败 , attempt.getAttemptNumber());  } } ) 图片 图片 总结 spring-retry 和 guava-retry 工具都是线程安全的重试能够支持并发业务场景的重试逻辑正确性。两者都很好的将正常方法和重试方法进行了解耦可以设置超时时间、重试次数、间隔时间、监听结果、都是不错的框架。 但是明显感觉得到guava-retry在使用上更便捷更灵活能根据方法返回值来判断是否重试而Spring-retry只能根据抛出的异常来进行重试。
http://www.huolong8.cn/news/90654/

相关文章:

  • 那些网站可以做公司的推广山东住房和城乡建设厅网站首页
  • 大庆做网站找谁网络营销产品价格策略
  • 专做蓝领招聘网站有哪些免费空间服务的网站
  • 专业自助建站图书馆网站建设的意义
  • 手机网站菜单设计模板html网站建设实例代码
  • 网站所有页面扬州互联网公司
  • 什么网站能看男女做暧公司注册代理中介
  • 网站建设经营服务合同范本用wordpress做广告收益
  • 豫建设标 网站重庆搜索排名提升
  • 莱西做网站国内重大新闻事件
  • 进度跟踪网站开发wordpress hq
  • 外贸电子网站上海贸易网站建设
  • 苏州产品网站建设怎么做网站镜像
  • 好用的ppt模板免费下载网站莱州网监局
  • 深圳建站公司外围网站字体加载不出来怎么办
  • 导航 网站 分析深圳南山做网站的公司
  • 台州市建设工程质量检测中心网站网站模板 使用
  • 安阳住房与城乡建设局官方网站西宁做网站最好的公司哪家好
  • 威海企业网站建设vs 2012网站开发
  • 网站关闭与域名备案网站开发公司合作协议书
  • 网站关键词优化的步骤和过程网站建设哪家好万维科技
  • 进度环wordpress无锡seo关键词排名
  • 阳江网站seo公司做网站设计网站建设推广
  • 备案的网站名称可以改吗电子商务公司网站模版
  • 做网站具体步骤成都sem优化
  • 广州市网站建设分站价格安陆网站建设
  • 线上职业技能培训平台吉林seo基础知识
  • 网站经营许可备案号网站建站收费
  • 做网站去除视频广告郑州做网站优化的公司
  • 湛江制作网站学校自助建网站平台怎么收费