这篇文章主要介绍如何实现回调和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
具体看下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
以上是“如何实现回调和spring的LambdaSafe类”这篇文章的所有内容,感谢各位的阅读!希望分享的内容对大家有帮助,更多相关知识,欢迎关注亿速云行业资讯频道!
亿速云「云服务器」,即开即用、新一代英特尔至强铂金CPU、三副本存储NVMe SSD云盘,价格低至29元/月。点击查看>>
免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。
原文链接:http://blog.itpub.net/31559758/viewspace-2673199/