热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

@autowired报错如何解决_踩坑了!使用@Autowired注入成功,GetBean方法却获取不到?!...

踩坑了?!之前推文已经讲过当Transactional遇到CacheEvict,你的代码是不是有bug!现在要在事务提交之后

踩坑了?!

之前推文已经讲过 当@Transactional遇到@CacheEvict,你的代码是不是有bug! 现在要在事务提交之后清除缓存。在 Spring4.2 之后,可以使用@TransactionalEventListener选择在事务提交之后再消费对应的事件。

b4b6510ff3b4b193fd915656e4cd993e.png
@TransactionalEventListener

为了方便发送事件,偷懒使用了静态方法:

6873254bfdab0078f0b1363188189d3e.png
publishEven

其中,SpringUtil.getBean()方法的内部实现为:

acd2dd1a2d11b1d6e86693ab0b67e210.png
getBean

满心欢喜写完代码,一运行,直接报错,报错信息为 IoC 容器中不存在ApplicationEventPublisher

怎么解决?解决的方案也很简单:使用@Autowired注入ApplicationEventPublisher,调用其publishEvent方法。

深入思考

可是,为什么ApplicationEventPublisher可以通过@Autowired进行注入,却不能使用BeanFactory#getBean方法来获取呢?

画外音:千万不要只限于解决问题,多思考,知其然并知其所以然。

看过 Spring 源码的小伙伴不知道还记不记得,在refresh()方法中,会调用prepareBeanFactory,在该方法中,注册了可解析依赖项。

85d0e7d80c9a4324fd855097735acc6e.png
image.png

从源码中可以知道,一些特殊实例对象是存放在DefaultListableBeanFactory#resolvableDependencies变量中的,在容器启动时,如果发现需要注入这些特定的实例对象,就直接在该变量中获取,自然也就不能通过BeanFactory#getBean方法来获取了。

DEMO

写个 demo 程序来尝试使用一下resolvableDependencies

8bace67b8bb472bbac054ae01bb9f708.png
demo

测试结果和ApplicationEventPublisher一样,可以在 Bean 中通过@Autowired的方式来注入MySpecificBean,但却无法通过BeanFactory#getBean方法来获取。

8e686b9f515e22d4eb263ac400041e48.png
test

写在最后

很多人都说要看源码,但是看源码的目的是什么?看源码,不是为了面试吹牛,也不是为了重复造轮子。看源码是为了深入了解这个框架的底层原理,为了以后遇到问题不会懵,可以解决百度无法解决的问题。我们可以学习优秀源码的设计思想,学习如何取一个好的类名、好的方法名,如何使用设计模式等等。

最后,小黑才疏学浅,文中不足之处还望你不吝斧正,感谢你的阅读~

a41c8ac2d3d0ef4ecde47176af61f53d.png



推荐阅读
author-avatar
mobiledu2502875213
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有