【Spring】BeanName 的自动生成原理

2023-09-17 12:42:14

🎈博客主页:🌈我的主页🌈
🎈欢迎点赞 👍 收藏 🌟留言 📝 欢迎讨论!👏
🎈本文由 【泠青沼~】 原创,首发于 CSDN🚩🚩🚩
🎈由于博主是在学小白一枚,难免会有错误,有任何问题欢迎评论区留言指出,感激不尽!🌠个人主页



🌟 一、默认 name 生成原理

在 Spring 中,提供了 BeanNameGenerator 用来生成 BeanName:

public interface BeanNameGenerator {

	/**
	 * Generate a bean name for the given bean definition.
	 * @param definition the bean definition to generate a name for
	 * @param registry the bean definition registry that the given definition
	 * is supposed to be registered with
	 * @return the generated bean name
	 */
	String generateBeanName(BeanDefinition definition, BeanDefinitionRegistry registry);

}

在这里插入图片描述

  • DefaultBeanNameGenerator:XML 配置中,默认的 BeanName 就是在这个中自动生成的

  • AnnotationBeanNameGenerator:Java 配置中,如果使用了 @Component 等注解标记的 Bean,没有设置默认的名称,则通过这个来生成默认的 BeanName

public class DefaultBeanNameGenerator implements BeanNameGenerator {

	/**
	 * A convenient constant for a default {@code DefaultBeanNameGenerator} instance,
	 * as used for {@link AbstractBeanDefinitionReader} setup.
	 * @since 5.2
	 */
	public static final DefaultBeanNameGenerator INSTANCE = new DefaultBeanNameGenerator();


	@Override
	public String generateBeanName(BeanDefinition definition, BeanDefinitionRegistry registry) {
		return BeanDefinitionReaderUtils.generateBeanName(definition, registry);
	}

}

可以看到,generateBeanName 这个方法实际上代理了 BeanDefinitionReaderUtils.generateBeanName 方法的执行,真正的 BeanName 的生成是在这个方法中完成的
在这里插入图片描述

public static String generateBeanName(
		BeanDefinition definition, BeanDefinitionRegistry registry, boolean isInnerBean)
		throws BeanDefinitionStoreException {
    // 这里就是获取到 XML 中 bean 标签里边配置的 class 属性的值
	String generatedBeanName = definition.getBeanClassName();
    // 判断是否有 class 这个属性值,如果没有的话,则在 parnetName 存在的情况下,
    // 使用 parentName+$child 来作为 生成的 beanName
	if (generatedBeanName == null) {
		if (definition.getParentName() != null) {
			generatedBeanName = definition.getParentName() + "$child";
		}
        // 如果没有 parentName,则尝试使用 factoryBeanName
		else if (definition.getFactoryBeanName() != null) {
			generatedBeanName = definition.getFactoryBeanName() + "$created";
		}
	}
    // 如果经过上面的处理,还是没有 generatedBeanName,那么就要抛异常了
	if (!StringUtils.hasText(generatedBeanName)) {
		throw new BeanDefinitionStoreException("Unnamed bean definition specifies neither " +
				"'class' nor 'parent' nor 'factory-bean' - can't generate bean name");
	}
	if (isInnerBean) {
		// Inner bean: generate identity hashcode suffix.
		return generatedBeanName + GENERATED_BEAN_NAME_SEPARATOR + ObjectUtils.getIdentityHexString(definition);
	}
	// Top-level bean: use plain class name with unique suffix if necessary.
    // 我们的默认 BeanName,实际上是在这个方法中生成的
	return uniqueBeanName(generatedBeanName, registry);
}
public static String uniqueBeanName(String beanName, BeanDefinitionRegistry registry) {
	String id = beanName;
	int counter = -1;
	// Increase counter until the id is unique.
    //GENERATED_BEAN_NAME_SEPARATOR 实际上就是 #
    // 所有这里是把类的全路径和 # 拼在一起
	String prefix = beanName + GENERATED_BEAN_NAME_SEPARATOR;
    // 后面的判断表示这个 id 是否已经被注册了,如果已经被注册,则继续生成新的 id
	while (counter == -1 || registry.containsBeanDefinition(id)) {
		counter++;
		id = prefix + counter;
	}
    //最终生成的 id 就是 org.javaboy.bean.User#0
	return id;
}

由此可以看到,默认的 BeanName 就是类的全路径+#+序列号,如 com.dong.Cat#0com.dong.Cat#1
对于序列号为 0 的 BeanName,还有一个默认的名称,就是类的全路径,不加任何序列号
上面这个生成 BeanName 的方法是在 BeanDefinitionParserDelegate#parseBeanDefinitionElement 方法中执行的,具体的逻辑如下:

if (beanDefinition != null) {
    // 当前没有配置 BeanName,即 bean 标签中没有 id 或者 name 属性
	if (!StringUtils.hasText(beanName)) {
		try {
			if (containingBean != null) {
				beanName = BeanDefinitionReaderUtils.generateBeanName(
						beanDefinition, this.readerContext.getRegistry(), true);
			}
			else {
                //这个地方,最终会调用到上面的逻辑去生成 BeanName
                //com.dong.Cat#0
				beanName = this.readerContext.generateBeanName(beanDefinition);
				// Register an alias for the plain bean class name, if still possible,
				// if the generator returned the class name plus a suffix.
				// This is expected for Spring 1.2/2.0 backwards compatibility.
                // 获取一个类的全路径 com.dong.Cat
                //!this.readerContext.getRegistry().isBeanNameInUse(beanClassName) 表示 beanClassName 还没有作为一个 BeanName 注册到 Spring 容器中
				String beanClassName = beanDefinition.getBeanClassName();
				if (beanClassName != null &&
						beanName.startsWith(beanClassName) && beanName.length() > beanClassName.length() &&
						!this.readerContext.getRegistry().isBeanNameInUse(beanClassName)) {
					//将之添加别名中,相当于类的全路径本身,成为了 Bean 的一个别名
                    aliases.add(beanClassName);
				}
			}
			if (logger.isTraceEnabled()) {
				logger.trace("Neither XML 'id' nor 'name' specified - " +
						"using generated bean name [" + beanName + "]");
			}
		}
		catch (Exception ex) {
			error(ex.getMessage(), ele);
			return null;
		}
	}
	String[] aliasesArray = StringUtils.toStringArray(aliases);
	return new BeanDefinitionHolder(beanDefinition, beanName, aliasesArray);
}

这就是为什么默认生成的 BeanName 中,#0可有可无的原因

🌟 二、id 和 name 属性处理原理

id 和 name 属性的处理其实也是在 BeanDefinitionParserDelegate#parseBeanDefinitionElement 方法中:


// 获取 bean 标签中的 id 属性值,user
String id = ele.getAttribute(ID_ATTRIBUTE);
// 获取 bean 标签中 name 属性值,user;user2;user3
String nameAttr = ele.getAttribute(NAME_ATTRIBUTE);
List<String> aliases = new ArrayList<>();
if (StringUtils.hasLength(nameAttr)) {
    //MULTI_VALUE_ATTRIBUTE_DELIMITERS 变量实际上就是 ;,空格
    // 所以这个方法实际上就是根据 ; , 以及 空格 去拆分 nameAttr,将之拆分为一个数组
	String[] nameArr = StringUtils.tokenizeToStringArray(nameAttr, MULTI_VALUE_ATTRIBUTE_DELIMITERS);
	// name 拆出来的属性将作为这个 bean 的别名
    aliases.addAll(Arrays.asList(nameArr));
}
//使用 id 作为 beanName
String beanName = id;
//这里相当于判断这个 bean 标签没有 id 属性,但是有 name 属性
if (!StringUtils.hasText(beanName) && !aliases.isEmpty()) {
    //将 name 拆出来的集合中的第一项作为 beanName
	beanName = aliases.remove(0);
	if (logger.isTraceEnabled()) {
		logger.trace("No XML 'id' specified - using '" + beanName +
				"' as bean name and " + aliases + " as aliases");
	}
}

但是,经过上面的处理,beanName 还是有可能为空。如果还为空,则进入到上面的逻辑中,自动生成 BeanName

更多推荐

JavaEE初阶(5)多线程案例(定时器、标准库中的定时器、实现定时器、线程池、标准库中的线程池、实现线程池)

接上次博客:JavaEE初阶(4)(线程的状态、线程安全、synchronized、volatile、wait和notify、多线程的代码案例:单例模式——饿汉懒汉、阻塞队列)_di-Dora的博客-CSDN博客目录多线程案例定时器标准库中的定时器实现定时器线程池标准库中的线程池实现线程池多线程案例定时器定时器(Tim

Qt/C++音视频开发53-本地摄像头推流/桌面推流/文件推流/监控推流等

一、前言编写这个推流程序,最开始设计的时候是用视频文件推流,后面陆续增加了监控摄像头推流(其实就是rtsp视频流)、网络电台和视频推流(一般是rtmp或者http开头m3u8结尾的视频流)、本地摄像头推流(本地USB摄像头或者笔记本自带摄像头等)、桌面推流(将当前运行环境的系统桌面抓拍推流)。按照分类的话其实就是三大类

【Robotframework+python】实现http接口自动化测试

前言下周即将展开一个http接口测试的需求,刚刚完成的java类接口测试工作中,由于之前犯懒,没有提前搭建好自动化回归测试框架,以至于后期rd每修改一个bug,经常导致之前没有问题的case又产生了bug,所以需要一遍遍回归case,过程一直手工去执行,苦不堪言。所以,对于即将开始的http接口测试需求,立马花了两天时

数据分析:利用gpt建立双11活动的分析框架

promt:您是一名某电商平台的资深数据分析师,首先,您知道什么是双11活动吗?output:当然,双11活动,也称为“光棍节”购物节,是中国最大的在线购物节之一。这个活动最初由阿里巴巴集团于2009年发起,并迅速吸引了其他电商平台的参与。双11活动通常在每年的11月11日进行,因此得名“双11”。这一天,各大电商平台

mybatis-plus

1.特征无侵入:只做增强不做改变,引入它不会对现有工程产生影响,如丝般顺滑损耗小:启动即会自动注入基本CURD,性能基本无损耗,直接面向对象操作强大的CRUD操作:内置通用Mapper、通用Service,仅仅通过少量配置即可实现单表大部分CRUD操作,更有强大的条件构造器,满足各类使用需求支持Lambda形式调用:通

Java面向对象(四)

提示:以下是本篇文章正文内容,下面案例可供参考一、JavaBean是什么?1.概念JavaBean是一种Java语言写成的可重用组件。所谓javaBean,是指符合如下标准的Java类:类是公共的有一个无参的公共的构造器有属性,且有对应的get、set方法2.代码:publicclassCustomer{private

Vue3 基础 – 快速上手 & 常用指令

1.在HTML网页中使用vue3的3个基本步骤a.通过script标签的src属性,在当前网页中全局引入vue3的脚本文件:<scriptsrc="https://unpkg.com/vue@3/dist/vue.global.js"></script>b.创建vue3的单页面应用程序实例://2.1从Vue对象中解构

【数据可视化】动态条形图Python代码实现

使用Python中的bar_chart_race_cn库创建动态条形图前言数据可视化在今天的数据分析和传达信息中起着至关重要的作用。动态条形图是一种强大的数据可视化工具,可以帮助我们展示随时间变化的数据趋势。本文将介绍如何使用Python编程语言中的bar_chart_race_cn库创建动态条形图。动态条形图可以用于

JDBC实现数据库批量插入

目录一、JDBC实现批量插入几种方式二、PreparedStatementaddBatch方法使用三、Statement和PreparedStatement区别使用Java数据库连接(JDBC)实现批量插入可以提高数据库操作的效率,特别是在需要一次性插入多条数据时。一、JDBC实现批量插入几种方式使用PreparedS

Redis中是如何实现分布式锁的?

分布式锁常见的三种实现方式:数据库乐观锁;基于Redis的分布式锁;基于ZooKeeper的分布式锁。本次面试考点是,你对Redis使用熟悉吗?Redis中是如何实现分布式锁的。要点Redis要实现分布式锁,以下条件应该得到满足互斥性在任意时刻,只有一个客户端能持有锁。不能死锁客户端在持有锁的期间崩溃而没有主动解锁,也

实战 | 服务端开发与计算机网络结合的完美案例

前言大家好,我是Martin后端,可以说是仅次于算法岗之外竞争最为激烈的岗位,而其中的服务端开发也是很多人会选择在秋招中投递的一个岗位,我想对于很多人来说,走上服务端开发之路的起点就是一个回声服务器了。今天带大家实战一把,真实体验服务端底层数据交换的点点滴滴,在这过程中可以让你看见TCP三次握手四次挥手的具体过程,全程

热文推荐