共计 16620 个字符,预计需要花费 42 分钟才能阅读完成。
作者:小傅哥
博客:https://bugstack.cn
积淀、分享、成长,让本人和别人都能有所播种!😄
一、前言
你提出问题,就要给出解决方案!
最近有粉丝小伙伴反馈,与本人的下级沟通总是遇到阻碍,感觉不被了解。大部分时候他提出来的事件都可能会被领导说:“我没 get 到你的点”、“你想做的这个我的项目没有业务价值”、“你提出问题,就要给出解决方案”,等等诸如此类的答复。
鉴于具体情况要具体分析,可能咱们并不一定能判断出是谁的问题,导致在每次的交谈中呈现的一致。可能是 leader 有 leader 的苦衷和视角,也可能是员工有员工的了解和想法,所以最终没有达成统一。
但就带团队来讲,无效沟通很重要。就像:如果你说的都对,那我为什么和你争吵呢?与其压抑遇到的矛盾点,不如都摊开了聊,谁的视角和心怀更大,谁就多有一些同理心。
如果尖利的批评齐全隐没,温和的批评将会变得刺耳。
如果温和的批评也不被容许,缄默将被认为居心叵测。
如果缄默也不再容许,投诉不够卖命将是一种罪状。
如果只容许一种声音存在,那么,惟一存在的那个声音就是谎话。
二、面试题
谢飞机,小记!
,总感觉 Spring 也没啥看的,怎么面试官一问就能问出花?
面试官:Spring 的 getBean 中,transformedBeanName 的作用是什么?
谢飞机:不晓得呀,看单词意思如同是扭转 Bean 名称。
面试官:那这么说,你的 Bean 如果有 alias 别名,Spring 在获取 Bean 时候要怎么解决?
谢飞机:这!
面试官:那如果用了 depends-on 呢?
谢飞机:啊,我没用过 depends-on 我不晓得!
面试官:那你调试代码时候,看见过 BeanName 后面有 & 的状况吗,为啥会呈现?
谢飞机:我不配晓得!再见!
三、Bean 的获取过程
对于刚接触看 Spring 源码的搭档来说,可能很纳闷于怎么就获取一个 Bean 就这么多流程呢?
- 可能有 Bean 可能有别名、可能有依赖、也可能是被 BeanFactory 包装过,所以会有 transformedBeanName 来解决这些差异化行为。
- 有没有循环依赖、有没有父工厂、是单例还是原型、是懒加载还是预加载、在不在缓冲区,所以就有各种组合判断来做不同的流程。
- 提前暴漏对象、三级缓存、后置标记分明,所有的优化解决都是为了让整个 Bean 的获取更加高效。
所以,它为了适应各类的需要,变得越来越简单了。而这部分常识的深刻学习相对不只是为了应酬八股文,更多的是思考到在日常的 Spring 应用中遇到简单问题时有没有一个大抵通晓的流程,能够疾速定位问题,以及此类需要的技术实现计划是否能在当前的利用开发中起到肯定的指导作用,因为它是一种设计方案的具体实现。
1. getBean 外围流程图
- 整张图就是 getBean 过程中波及到的类和外围流程用到的办法以及操作的内容。如果你能把整张图全了解了,那么根本也就看懂了 getBean 的全过程。
- 本张图可能会因为网络压缩变得不清晰,能够通过 关注公众号 :bugstack 虫洞栈,回复:
图稿
,获取。
接下来,咱们就顺次的把对于获取 Bean 实例的重点代码列举进去做剖析,读者搭档也能够联合流程图一起看,这样会更不便了解。
2. getBean 从哪开始读源码
@Test
public void test_getBean() {BeanFactory beanFactory = new ClassPathXmlApplicationContext("spring-config.xml");
UserDao userDao = beanFactory.getBean("userDao", UserDao.class);
logger.info("获取 Bean:{}", userDao);
}
- 在日常利用到 Spring 的开发中根本都是基于注解,简直不会本人去应用
beanFactory.getBean
的形式去获取一个 Bean 实例。 - 所以在你学习的时候如果找不到查看 getBean 源码的入口,也不不便调试相熟源码时,能够写这样一个单元测试类,点入到 getBean 就能够浏览源码了。
3. getBean 源码全局预览
源码地位:AbstractBeanFactory -> getBean() -> doGetBean()
@Override
public <T> T getBean(String name, Class<T> requiredType) throws BeansException {
// getBean 就像你的领导其实没做啥,都在 doGetBean 里
return doGetBean(name, requiredType, null, false);
}
protected <T> T doGetBean(final String name, final Class<T> requiredType, final Object[] args, boolean typeCheckOnly)
throws BeansException {
// 解决别名 BeanName、解决带 & 符的工厂 BeanName
final String beanName = transformedBeanName(name);
Object bean;
// 先尝试从缓存中获取 Bean 实例,这个地位就是三级缓存解决循环依赖的办法
Object sharedInstance = getSingleton(beanName);
if (sharedInstance != null && args == null) {if (logger.isDebugEnabled()) {if (isSingletonCurrentlyInCreation(beanName)) {
logger.debug("Returning eagerly cached instance of singleton bean'" + beanName +
"'that is not fully initialized yet - a consequence of a circular reference");
}
else {logger.debug("Returning cached instance of singleton bean'" + beanName + "'");
}
}
// 1. 如果 sharedInstance 是一般的 Bean 实例,则上面的办法会间接返回
// 2. 如果 sharedInstance 是工厂 Bean 类型,则须要获取 getObject 办法,能够参考对于 FactoryBean 的实现类
bean = getObjectForBeanInstance(sharedInstance, name, beanName, null);
}
else {
// 循环依赖有三种,setter 注入、多实例和构造函数,Spring 只能解决 setter 注入,所以这里是 Prototype 则会抛出异样
if (isPrototypeCurrentlyInCreation(beanName)) {throw new BeanCurrentlyInCreationException(beanName);
}
// 1. 父 bean 工厂存在
// 2. 以后 bean 不存在于以后 bean 工厂,则到父工厂查找 bean 实例
BeanFactory parentBeanFactory = getParentBeanFactory();
if (parentBeanFactory != null && !containsBeanDefinition(beanName)) {
// 获取 name 对应的 beanName,如果 name 是以 & 结尾,则返回 & + beanName
String nameToLookup = originalBeanName(name);
// 依据 args 参数是否为空,调用不同的父容器办法获取 bean 实例
if (args != null) {return (T) parentBeanFactory.getBean(nameToLookup, args);
}
else {return parentBeanFactory.getBean(nameToLookup, requiredType);
}
}
// 1. typeCheckOnly,用于判断调用 getBean 办法时,是否仅是做类型查看
// 2. 如果不是只做类型查看,就会调用 markBeanAsCreated 进行记录
if (!typeCheckOnly) {markBeanAsCreated(beanName);
}
try {
// 从容器 getMergedLocalBeanDefinition 获取 beanName 对应的 GenericBeanDefinition,转换为 RootBeanDefinition
final RootBeanDefinition mbd = getMergedLocalBeanDefinition(beanName);
// 查看以后创立的 bean 定义是否为形象 bean 定义
checkMergedBeanDefinition(mbd, beanName, args);
// 解决应用了 depends-on 注解的依赖创立 bean 实例
String[] dependsOn = mbd.getDependsOn();
if (dependsOn != null) {for (String dep : dependsOn) {
// 监测是否存在 depends-on 循环依赖,若存在则会抛出异样
if (isDependent(beanName, dep)) {throw new BeanCreationException(mbd.getResourceDescription(), beanName,
"Circular depends-on relationship between'" + beanName + "'and'" + dep + "'");
}
// 注册依赖记录
registerDependentBean(dep, beanName);
try {
// 加载 depends-on 依赖(dep 是 depends-on 缩写)getBean(dep);
}
catch (NoSuchBeanDefinitionException ex) {throw new BeanCreationException(mbd.getResourceDescription(), beanName,
"'"+ beanName +"' depends on missing bean '"+ dep +"'", ex);
}
}
}
// 创立单例 bean 实例
if (mbd.isSingleton()) {
// 把 beanName 和 new ObjectFactory 匿名外部类传入回调
sharedInstance = getSingleton(beanName, new ObjectFactory<Object>() {
@Override
public Object getObject() throws BeansException {
try {
// 创立 bean
return createBean(beanName, mbd, args);
}
catch (BeansException ex) {
// 创立失败则销毁
destroySingleton(beanName);
throw ex;
}
}
});
bean = getObjectForBeanInstance(sharedInstance, name, beanName, mbd);
}
// 创立其余类型的 bean 实例
else if (mbd.isPrototype()) {
// It's a prototype -> create a new instance.
Object prototypeInstance = null;
try {beforePrototypeCreation(beanName);
prototypeInstance = createBean(beanName, mbd, args);
}
finally {afterPrototypeCreation(beanName);
}
bean = getObjectForBeanInstance(prototypeInstance, name, beanName, mbd);
}
else {String scopeName = mbd.getScope();
final Scope scope = this.scopes.get(scopeName);
if (scope == null) {throw new IllegalStateException("No Scope registered for scope name'" + scopeName + "'");
}
try {Object scopedInstance = scope.get(beanName, new ObjectFactory<Object>() {
@Override
public Object getObject() throws BeansException {beforePrototypeCreation(beanName);
try {return createBean(beanName, mbd, args);
}
finally {afterPrototypeCreation(beanName);
}
}
});
bean = getObjectForBeanInstance(scopedInstance, name, beanName, mbd);
}
catch (IllegalStateException ex) {
throw new BeanCreationException(beanName,
"Scope'" + scopeName + "'is not active for the current thread; consider" +
"defining a scoped proxy for this bean if you intend to refer to it from a singleton",
ex);
}
}
}
catch (BeansException ex) {cleanupAfterBeanCreationFailure(beanName);
throw ex;
}
}
// 如果须要类型转换,这里会进行操作
if (requiredType != null && bean != null && !requiredType.isInstance(bean)) {
try {return getTypeConverter().convertIfNecessary(bean, requiredType);
}
catch (TypeMismatchException ex) {if (logger.isDebugEnabled()) {
logger.debug("Failed to convert bean'" + name + "'to required type'" +
ClassUtils.getQualifiedName(requiredType) + "'", ex);
}
throw new BeanNotOfRequiredTypeException(name, requiredType, bean.getClass());
}
}
// 返回 Bean
return (T) bean;
}
综上根本就是 getBean 过程波及到的外围解决办法,根本包含;
- transformedBeanName,解决别名 BeanName、解决带 & 符的工厂 BeanName。
- getSingleton,先尝试从缓存中获取 Bean 实例,这个地位就是三级缓存解决循环依赖的办法。
- getObjectForBeanInstance,如果 sharedInstance 是一般的 Bean 实例,则上面的办法会间接返回。另外 sharedInstance 是工厂 Bean 类型,则须要获取 getObject 办法,能够参考对于 FactoryBean 的实现类。
- isPrototypeCurrentlyInCreation,循环依赖有三种,setter 注入、多实例和构造函数,Spring 只能解决 setter 注入,所以这里是 Prototype 则会抛出异样。
- getParentBeanFactory,父 bean 工厂存在,以后 bean 不存在于以后 bean 工厂,则到父工厂查找 bean 实例。
- originalBeanName,获取 name 对应的 beanName,如果 name 是以 & 结尾,则返回 & + beanName
- args != null,依据 args 参数是否为空,调用不同的父容器办法获取 bean 实例
- !typeCheckOnly,typeCheckOnly,用于判断调用 getBean 办法时,是否仅是做类型查看,如果不是只做类型查看,就会调用 markBeanAsCreated 进行记录
- mbd.getDependsOn,解决应用了 depends-on 注解的依赖创立 bean 实例
- isDependent,监测是否存在 depends-on 循环依赖,若存在则会抛出异样
- registerDependentBean,注册依赖记录
- getBean(dep),加载 depends-on 依赖(dep 是 depends-on 缩写)
- mbd.isSingleton(),创立单例 bean 实例
- mbd.isPrototype(),创立其余类型的 bean 实例
- return (T) bean,返回 Bean 实例
4. beanName 转换操作
解决 & 符:transformedBeanName() -> BeanFactoryUtils.transformedBeanName(name)
public static String transformedBeanName(String name) {Assert.notNull(name, "'name' must not be null");
String beanName = name;
while (beanName.startsWith(BeanFactory.FACTORY_BEAN_PREFIX)) {beanName = beanName.substring(BeanFactory.FACTORY_BEAN_PREFIX.length());
}
return beanName;
}
- 应用 FactoryBean 创立出的对象,会在 DefaultListableBeanFactory 初始化的时候,应用 getBean(FACTORY_BEAN_PREFIX + beanName) 给 beanName 加上 &
(String FACTORY_BEAN_PREFIX = "&")
- 这里是应用 while 循环逐渐的把 & 去掉,只有截取首个字符是 & 符,就持续循环截取。
&&&userService -> &&userService -> &userService -> userService
别名转换:transformedBeanName() -> canonicalName
public String canonicalName(String name) {
String canonicalName = name;
// Handle aliasing...
String resolvedName;
do {resolvedName = this.aliasMap.get(canonicalName);
if (resolvedName != null) {canonicalName = resolvedName;}
}
while (resolvedName != null);
return canonicalName;
}
<bean id="userService" class="org.itstack.interview.UserService"/>
<alias name="userService" alias="userService-alias01"/>
<alias name="userService-alias01" alias="userService-alias02"/>
- 首先 Spring 对 Bean 的寄存并不会应用别名作为 Map 中的 key,所以遇到所有别名获取 Bean 都须要查到对应原来名字,才能够。如果你晓得这个事,是不遇到此类问题时,就晓得从哪下手查了
- do…while 循环会顺次像链条一样一直的寻找别名对应的名称,直到以后这个名称没有别名了,就返回对应 BeanName
5. depends-on 依赖 Bean
AbstractBeanFactory -> isDependent(beanName, dep) -> DefaultSingletonBeanRegistry
protected boolean isDependent(String beanName, String dependentBeanName) {synchronized (this.dependentBeanMap) {return isDependent(beanName, dependentBeanName, null);
}
<bean id="userService" class="org.itstack.interview.UserService" depends-on="userDao"/>
<bean id="userDao" class="org.itstack.interview.UserDao"/>
- isDependent 解决的是应用了 depends-on 配置的 Bean 定义。
private boolean isDependent(String beanName, String dependentBeanName, Set<String> alread
if (alreadySeen != null && alreadySeen.contains(beanName)) {return false;}
String canonicalName = canonicalName(beanName);
Set<String> dependentBeans = this.dependentBeanMap.get(canonicalName);
if (dependentBeans == null) {return false;}
if (dependentBeans.contains(dependentBeanName)) {return true;}
for (String transitiveDependency : dependentBeans) {if (alreadySeen == null) {alreadySeen = new HashSet<String>();
}
alreadySeen.add(beanName);
if (isDependent(transitiveDependency, dependentBeanName, alreadySeen)) {return true;}
}
return false;
}
- alreadySeen != null,监测曾经依赖的 Bean
- canonicalName,解决别名配置,找到最原来是的 BeanName
- Set<String> dependentBeans,获取依赖的 Bean 汇合
- for 循环递归检测依赖的 Bean,并增加到 alreadySeen 中
AbstractBeanFactory -> registerDependentBean(dep, beanName) -> DefaultSingletonBeanRegistry
public void registerDependentBean(String beanName, String dependentBeanName) {String canonicalName = canonicalName(beanName);
synchronized (this.dependentBeanMap) {Set<String> dependentBeans = this.dependentBeanMap.get(canonicalName);
if (dependentBeans == null) {dependentBeans = new LinkedHashSet<String>(8);
this.dependentBeanMap.put(canonicalName, dependentBeans);
}
dependentBeans.add(dependentBeanName);
}
synchronized (this.dependenciesForBeanMap) {
Set<String> dependenciesForBean = this.dependenciesForBeanMap.get(dependentBeanName
if (dependenciesForBean == null) {dependenciesForBean = new LinkedHashSet<String>(8);
this.dependenciesForBeanMap.put(dependentBeanName, dependenciesForBean);
}
dependenciesForBean.add(canonicalName);
}
}
- canonicalName(beanName),获取原始的 beanName
- synchronized (this.dependentBeanMap),增加 <canonicalName, dependentBeanName> 到 dependentBeanMap 中
- synchronized (this.dependenciesForBeanMap),增加 <dependentBeanName, canonicalName> 到 dependenciesForBeanMap 中
最初:getBean(dep),就能够获取到 depends-on 依赖的 Bean 了
6. 解决单实例 Bean
AbstractBeanFactory -> mbd.isSingleton()
if (mbd.isSingleton()) {sharedInstance = getSingleton(beanName, new ObjectFactory<Object>() {
@Override
public Object getObject() throws BeansException {
try {return createBean(beanName, mbd, args);
}
catch (BeansException ex) {destroySingleton(beanName);
throw ex;
}
}
});
bean = getObjectForBeanInstance(sharedInstance, name, beanName, mbd);
}
- 这一部分是应用 beanName 和 singletonFactory 匿名外部类传入期待回调的形式创立单实例 Bean 实例
public Object getSingleton(String beanName, ObjectFactory<?> singletonFactory) {Assert.notNull(beanName, "'beanName' must not be null");
synchronized (this.singletonObjects) {Object singletonObject = this.singletonObjects.get(beanName);
if (singletonObject == null) {if (this.singletonsCurrentlyInDestruction) {
throw new BeanCreationNotAllowedException(beanName,
"Singleton bean creation not allowed while singletons of this factory are in destruction" +
"(Do not request a bean from a BeanFactory in a destroy method implementation!)");
}
if (logger.isDebugEnabled()) {logger.debug("Creating shared instance of singleton bean'" + beanName + "'");
}
beforeSingletonCreation(beanName);
boolean newSingleton = false;
boolean recordSuppressedExceptions = (this.suppressedExceptions == null);
if (recordSuppressedExceptions) {this.suppressedExceptions = new LinkedHashSet<Exception>();
}
try {singletonObject = singletonFactory.getObject();
newSingleton = true;
}
catch (IllegalStateException ex) {singletonObject = this.singletonObjects.get(beanName);
if (singletonObject == null) {throw ex;}
}
catch (BeanCreationException ex) {if (recordSuppressedExceptions) {for (Exception suppressedException : this.suppressedExceptions) {ex.addRelatedCause(suppressedException);
}
}
throw ex;
}
finally {if (recordSuppressedExceptions) {this.suppressedExceptions = null;}
afterSingletonCreation(beanName);
}
if (newSingleton) {addSingleton(beanName, singletonObject);
}
}
return (singletonObject != NULL_OBJECT ? singletonObject : null);
}
}
- this.singletonObjects.get(beanName),先尝试从缓存池中获取对象,没有就持续往下执行
- beforeSingletonCreation(beanName),标记以后 bean 被创立,如果有构造函数注入的循环依赖会报错
- singletonObject = singletonFactory.getObject(),创立 bean 过程就是调用 createBean() 办法
- afterSingletonCreation(beanName),最初把标记从汇合中移除
- addSingleton(beanName, singletonObject),新创建的会退出缓存汇合
7. 从缓存中获取 bean 实例
doCreateBean -> if (earlySingletonExposure) -> getSingleton(beanName, false)
protected Object getSingleton(String beanName, boolean allowEarlyReference) {
// 从 singletonObjects 获取实例,singletonObjects 中缓存的实例都是齐全实例化好的 bean,能够间接应用
Object singletonObject = this.singletonObjects.get(beanName);
// 如果 singletonObject 为空,则没有创立或创立中
if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) {
// 加锁
synchronized (this.singletonObjects) {
// 单例缓存池中,没有以后 beanName
singletonObject = this.earlySingletonObjects.get(beanName);
if (singletonObject == null && allowEarlyReference) {ObjectFactory<?> singletonFactory = this.singletonFactories.get(beanName);
if (singletonFactory != null) {
// 退出到三级缓存,暴漏晚期对象用于解决三级缓存
singletonObject = singletonFactory.getObject();
this.earlySingletonObjects.put(beanName, singletonObject);
this.singletonFactories.remove(beanName);
}
}
}
}
return (singletonObject != NULL_OBJECT ? singletonObject : null);
}
- 其实这一段代码次要就是应用三级缓存解决 set 注入循环依赖的,前面会独自列一个章节对循环依赖做相干试验验证
- singletonObjects,用于寄存初始化好的 bean 实例。
- earlySingletonObjects,用于寄存初始化中的 bean,来解决循环依赖。
-
singletonFactories,用于寄存 bean 工厂,bean 工厂所生成的 bean 还没有实现初始化 bean。
8. FactoryBean 中获取 bean 实例
AbstractBeanFactory -> getObjectForBeanInstance(sharedInstance, name, beanName, null)
protected Object getObjectForBeanInstance(Object beanInstance, String name, String beanName, RootBeanDefinition mbd) {
// 如果 beanName 以 & 结尾,但 beanInstance 却不是 FactoryBean,则会抛出异样
if (BeanFactoryUtils.isFactoryDereference(name) && !(beanInstance instanceof FactoryBean)) {throw new BeanIsNotAFactoryException(transformedBeanName(name), beanInstance.getClass());
}
// 这里判断就是这个 bean 是不是 FactoryBean,不是就间接返回了
if (!(beanInstance instanceof FactoryBean) || BeanFactoryUtils.isFactoryDereference(name)) {return beanInstance;}
Object object = null;
if (mbd == null) {
// 如果 mbd 为空,则从缓存加载 bean(FactoryBean 生成的单例 bean 实例会缓存到 factoryBeanObjectCache 汇合中,方便使用)object = getCachedObjectForFactoryBean(beanName);
}
if (object == null) {
// 到这,beanInstance 是 FactoryBean 类型,所以就强转了
FactoryBean<?> factory = (FactoryBean<?>) beanInstance;
// mbd 为空且判断 containsBeanDefinition 是否蕴含 beanName
if (mbd == null && containsBeanDefinition(beanName)) {
// 合并 BeanDefinition
mbd = getMergedLocalBeanDefinition(beanName);
}
boolean synthetic = (mbd != null && mbd.isSynthetic());
// 调用 getObjectFromFactoryBean 获取实例
object = getObjectFromFactoryBean(factory, beanName, !synthetic);
}
return object;
}
- (!(beanInstance instanceof FactoryBean),这里判断就是这个 bean 是不是 FactoryBean,不是就间接返回了
- 如果 mbd 为空,则从缓存加载 bean(FactoryBean 生成的单例 bean 实例会缓存到 factoryBeanObjectCache 汇合中,方便使用)
- 调用 getObjectFromFactoryBean 获取实例,这里会包含一部分对单例以及非单例的解决,以及最终返回 factory.getObject(); 对应的 Bean 实例
四、测试案例
1. 别名
<bean id="userService" class="org.itstack.interview.UserService"/>
<!-- 起个别名 -->
<alias name="userService" alias="userService-alias01"/>
<!-- 别名的别名 -->
<alias name="userService-alias01" alias="userService-alias02"/>
@Test
public void test_alias() {BeanFactory beanFactory = new ClassPathXmlApplicationContext("spring-config-alias.xml");
UserService userService = beanFactory.getBean("userService-alias02", UserService.class);
logger.info("获取 Bean 通过别名:{}", userService);
}
- 在单元测试 getBean 的时候,会看到它会把别名逐渐解决掉,最终获取到原有的 BeanName
2. 依赖
<bean id="userService" class="org.itstack.interview.UserService" depends-on="userDao"/>
<bean id="userDao" class="org.itstack.interview.UserDao"/>
@Test
public void test_depends_on() {BeanFactory beanFactory = new ClassPathXmlApplicationContext("spring-config-depends-on.xml");
UserService userService = beanFactory.getBean(UserService.class, "userService");
logger.info("获取 Bean:{}", userService.getUserDao());
}
- 波及到依赖会走到 dependsOn != null 下,获取到依赖的 Bean 实例。
3. BeanFactory
<bean id="userDao" class="org.itstack.interview.MyFactoryBean"/>
@Test
public void test_factory_bean() {BeanFactory beanFactory = new ClassPathXmlApplicationContext("spring-config-factory-bean.xml");
UserDao userDao = beanFactory.getBean("userDao", UserDao.class);
logger.info("获取 Bean:{}", userDao);
}
- 实现 FactoryBean 的类会须要实现 getObject 办法,所有此类的 Bean 最终都是获取 getObject
五、总结
- 到这里对于 Spring IOC 获取 Bean 的外围流程根本就全副介绍完了,整个篇章让咱们看到获取一个 Bean 的流程也是非常复杂的,波及到了十分多的分支流程。之所有会有这么多的流程,就是咱们后面介绍到的,因为 Spring 的 Bean 获取须要满足很多种状况。
- 在学习的过程能够优先依照 GetBean 流程图进行梳理,之后对照源码按步骤剖析,这样的过程简直会耗费你 1~2 天的工夫,但整个过程学习完,根本也就对 GetBean 没有什么生疏了。
- 学习简直就是一个缓缓磨的过程,就像走迷宫一样,尽管有时候会走错路,但那些错了的路也是常识学习的一部分。在编程的学习中不只是看后果,过程是更重要的,学会学习的形式更有意义。
六、系列举荐
- 程序员怎么开发一个 SpringBoot Starter?
- 你说,怎么把 Bean 塞到 Spring 容器?
- Spring IOC 个性有哪些?
- Thread.start(),它是怎么让线程启动的呢?-%E5%AE%83%E6%98%AF%E6%80%8E%E4%B9%88%E8%AE%A9%E7%BA%BF%E7%A8%8B%E5%90%AF%E5%8A%A8%E7%9A%84%E5%91%A2.html)
- 小伙伴美团一面的分享和剖析(含解答)