踩坑了!使用 @Autowired 注入成功,GetBean 方法却获取不到?!

本文首发于个人微信公众号:Coder小黑

踩坑了?!

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

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

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

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

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

深入思考

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

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

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

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

DEMO

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

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

写在最后

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

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



欢迎关注公众号,公众号全是个人原创文章,而且内容更多哦

原文地址:https://www.cnblogs.com/coderxiaohei/p/12047337.html

时间: 2024-08-29 09:49:27

踩坑了!使用 @Autowired 注入成功,GetBean 方法却获取不到?!的相关文章

每日踩坑 2019-08-22 Unity 依赖注入 未能为类选择构造函数

Unity会自动使用参数最多的构造函数来进行创建对象, 假如在这个类中有多个构造函数时,而我们要指定其中一个作为Unity进行创建对象则必需用到 [InjectionConstructor] 气死我了 怎么会有这么睿智的玩意, 明明都传 ParameterOverride 了, 居然不会根据参数匹配. 参考:https://www.cnblogs.com/wujy/p/3321478.html https://www.cnblogs.com/chinadeveloper/p/6184307.ht

MyBatis踩坑之SQLProvider转义字符被删除问题

目录 踩坑背景 问题描述 原因追踪 解决方案 方法一 方法二 踩坑背景 项目架构:Spring Boot + MyBatis + MySQL. 使用MyBatis作为ORM框架,jdbc驱动使用的是mariadb-java-client. <dependency> <groupId>org.mariadb.jdbc</groupId> <artifactId>mariadb-java-client</artifactId> <version

ApplicationContext之getBean方法详解

我们知道可以通过ApplicationContext的getBean方法来获取Spring容器中已初始化的bean.getBean一共有以下四种方法原型: l getBean(String name) l getBean(Class<T> type) l getBean(String name,Class<T> type) l getBean(String name,Object[] args) 下来我们分别来探讨以上四种方式获取bean的区别.其中实体类Person定义如下: p

Spring Boot 开发系列一 开发踩坑

这是学习spring boot 的第二周,公司号称这玩意是啥都不会的新手就可以填空开发,于是决定上手一把,怎么说我也是搞了快七八年的.NET和.NETcore,没想到无情打脸,快被这个能填空开的IDE搞疯了,下面是记下一些自己踩坑开发中遇到的一系列无穷无尽的问题. 一 .集成MyBatis a)首先如何理解这个  MyBatis 的东西,我的理解是 :MyBatis 有点类似ORM的感觉,跟 JPA很类似,就是一个orm,需要一个类似 dbcontext的东西,这个东西,在MyBatis 里面叫

Spring @Transactional踩坑记

@Transactional踩坑记 总述 ? Spring在1.2引入@Transactional注解, 该注解的引入使得我们可以简单地通过在方法或者类上添加@Transactional注解,实现事务控制. 然而看起来越是简单的东西,背后的实现可能存在很多默认规则和限制.而对于使用者如果只知道使用该注解,而不去考虑背后的限制,就可能事与愿违,到时候线上出了问题可能根本都找不出啥原因. 踩坑记 1. 多数据源 事务不生效 背景介绍 ? 由于数据量比较大,项目的初始设计是分库分表的.于是在配置文件中

利用vue-router和compoment重构代码--踩坑

业务主要功能 获取所有的数据库列表 点击某一个数据库列表的时候,右侧分页展示数据 点击右侧某一条数据的时候,现实数据详情 以下是之前的页面,存在以下问题: 前段开发没有工程化(webpack) 主要功能耦合,列表,详情,(检索,重构的是为了加功能方便) 左侧的数据库链接是直接跳页的,分页的链接是跳页的,右侧点击的详情页却是vue-resource加载的 代码结构混乱,之前为了快速实现功能.所有代码写在一个文件上.难读. 功能效果图 数据列表页效果 数据详情效果 代码重构思路 前段模块化开发,用w

阿里云磁盘扩容踩坑总结

公司半年前上线一个新的项目,采购了一批阿里云主机,磁盘组成是40G系统盘+100G的数据盘,数据库采用MariaDB Galera Cluster集群部署,由于业务数据量快速增长,导致磁盘存储空间剩余量很少,急需要扩容,先总结整个项目规划中埋下的坑: 1.没有DBA对数据库的容量规划,而前期的运维人员采购时选用100G的SSD云盘: 2.数据库默认使用共享表空间,缺点是删除数据后不释放空间,当数据快速增长后,我们采取了先删除临时表数据的方式来尽量避免暴力扩容,争取在春节期间稳定,删除部分数据后,

运维小白部署网站踩坑全过程

作为一名一个星期之前还是个运维小白的我,经过一番摸爬滚打终于把我的小网站部署上去了 一.服务器环境 OK,一开始我得到的是一台centos的服务器,没有装apache/nginx.mysql和php环境,一开始我是使用lnmp一键安装包,结果发现php好多扩展和依赖缺失,重装了php后才解决了问题,mysql也遇到一些问题,所以建议大家在装环境时除非一键lamp/lnmp安装包经历过实践的考验,不然还是老老实实地一个一个组件安装,php.mysql.nginx. lnmp/lamp安装成功的标志

jQuery版本升级踩坑大全

背景 -------------------------------------------------------------------------------- jQuery想必各个web工程师都再熟悉不过了,不过现如今很多网站还采用了很古老的jQuery版本.其实如果早期版本使用不当,可能会有DOMXSS漏洞,非常建议升级到jQuery 1.9.x或以上版本.前段时间我就主导了这件事情,把公司里我们组负责的项目jQuery版本从1.4.2升级到了jQuery 1.11.3.jQuery官