温馨提示×

温馨提示×

您好,登录后才能下订单哦!

密码登录×
登录注册×
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》

如何实现回调和spring的LambdaSafe类

发布时间:2021-12-14 16:54:21 来源:亿速云 阅读:242 作者:小新 栏目:编程语言

这篇文章主要介绍如何实现回调和spring的LambdaSafe类,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!

   在阅读spring boot源码时发现了WebServerFactoryCustomizerBeanPostProcessor类中有

LambdaSafe.callbacks(WebServerFactoryCustomizer.class, getCustomizers(), webServerFactory)
                .withLogger(WebServerFactoryCustomizerBeanPostProcessor.class)
                .invoke((customizer) -> customizer.customize(webServerFactory));

的用法。搜索了一下国内网站,发现没有相关资料,所以只能去spring官方网站上查阅文档学习一下。
   我们知道一切皆可“类”化的java中没有函数指针,而函数指针可以用来实现编程中较为普遍使用的“回调”。所谓“回调“是和“调用”相反的过程,如果用类和消息来理解就是,类A调用类B的方法,可以认为是类A向类B发送了一个消息,这个过程被称为”调用”。对于类B来说,B 被动 接收了这个消息。但是如果类B的方法在处理这个消息的过程中,又需要类A才有的数据或者类A才有能力拿到的数据,这时候类B需要向类A发送一个消息(调用类A的方法),主动 请求获取数据。这时候类B就需要知道类A的方法如何调用,这个过程称为”回调”。
   如果是c/c++,可以直接给类B传入一个类A方法的函数指针。但是java中不存在现成的“函数指针”可以使用,所以出现了几种类“函数指针”的形式。一种是比较常见的使用接口+方法的形式实现类函数指针的方式,这种接口被称为函数式接口,另一种是java 8中的新特性lambda表达式,本质上也是一种函数式接口。
   Java中定义了一个 @FunctionalInterface注解来修饰接口实现的”函数指针”,但是这个注解并不是必须的,只要满足函数式接口标准的接口都可以作为”函数指针”。如果一个接口不满足函数式接口标准,同时注解了 @FunctionalInterface,那么编译器就会报错,这个注解相当于能够更好的方便编译器进行检查。现在我们以上面代码中涉及的函数式接口来举例。

@FunctionalInterface
public interface WebServerFactoryCustomizer<T extends WebServerFactory> {
    void customize(T factory);
}
WebServerFactoryCustomizer是一个提供给用户用来定制WebServerFactory实例的接口。这个函数式接口的使用是在上面代码的invoke方法的lambda表达式中,主要是处理传入的WebServerFactory实例的bean,然后继承这个函数式接口就可以对这个实例为所欲为。
好了,基础介绍完了,终于到我们的主角LambdaSafe类了。spring 官方文档给的解释是

Utility that can be used to invoke lambdas in a safe way. Primarily designed to help support generically typed callbacks where class cast exceptions need to be dealt with due to class erasure.

。大意就是说LambdaSafe可以用来安全的运行lambda表达式,所谓的安全就是指可以支持处理在“回调”过程中因为类型擦除(java泛型中的概念)而需要抛出异常的情况。
   那么实际上如何实现的呢?先看LambdaSafe.callbacks方法。在上面代码中,LambdaSafe.callbacks传入了WebServerFactoryCustomizer.class、getCustomizers()和需要被为所欲为的WebServerFactory实例。getCustomizers()的返回值会获取所有WebServerFactoryCustomizer函数式接口的实现类,可以理解获取了一堆函数指针。

private List<WebServerFactoryCustomizer<?>> customizers;
private Collection<WebServerFactoryCustomizer<?>> getCustomizers() {
  if (this.customizers == null) {
    // Look up does not include the parent context
    this.customizers = new ArrayList<>(getWebServerFactoryCustomizerBeans());
    this.customizers.sort(AnnotationAwareOrderComparator.INSTANCE);
    this.customizers = Collections.unmodifiableList(this.customizers);
  }
  return this.customizers;
}

callbacks方法返回的是

public static <C, A> Callbacks<C, A>
callbacks(Class<C> callbackType,
                    Collection<? extends C> callbackInstances,
                    A argument,
                    Object... additionalArguments);

LambdaSafe中的内部类LambdaSafe.Callback,其中A和C是两个泛型,这里分别对应的是WebServerFactoryCustomizer.class和WebServerFactoryCustomizer.然后这个类有一个withLogger方法用来打印日志,这样在执行lambda表达式的过程中就能记录抛出的异常日志,并且日志头打印的是传入参数WebServerFactoryCustomizerBeanPostProcessor.class类(不会被擦除效果影响),最后这个withLogger方法返回的参数是SELF,代表LambdaSafe.Callback自己。这样LambdaSafe.Callback实例又调用了自己的invoke参数,执行了getCustomizers方法返回的所有WebServerFactoryCustomizer函数式接口的实现类的customize方法。
   具体看下invoke的代码,也挺有意思的。

private final Collection<? extends C> callbackInstances;
public void invoke(Consumer<C> invoker) {
  this.callbackInstances.forEach((callbackInstance) -> {
    invoke(callbackInstance, () -> {
      invoker.accept(callbackInstance);
      return null;
    });
  });
}

Consumer也是一种函数式接口,这里用来代表(customizer) -> customizer.customize(webServerFactory)的lambda表达式。callbackInstances存储了上面说到的所有WebServerFactoryCustomizer函数式接口的实现类,这里边遍历边通过invoker.accept()方法执行每个函数式接口的customize方法。这样整个代码逻辑就完全了。

以上是“如何实现回调和spring的LambdaSafe类”这篇文章的所有内容,感谢各位的阅读!希望分享的内容对大家有帮助,更多相关知识,欢迎关注亿速云行业资讯频道!

向AI问一下细节

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

AI