Spring源码(二-1)-XML文件的读取-bean标签

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
云解析 DNS,旗舰版 1个月
全局流量管理 GTM,标准版 1个月
简介: 上篇博文看到的四个标签中, bean 的解析最为复杂和重要。

bean标签的解析及注册

上篇博文看到的四个标签中, bean 的解析最为复杂和重要。进入DefaultBeanDefinitionDocumentReaderprocessBeanDefinition(ele, delegate)函数。

protected void processBeanDefinition(Element ele,
  BeanDefinitionParserDelegate delegate) {
  //委托 BeanDefinitionParserDelegate 类的 parseBeanDefinitionElement
  //方法进行元素解析,
  //返回 BeanDefinitionHolder对象,这个对象就会包含
  //BeanDefinition,beanName以及aliases
  BeanDefinitionHolder bdHolder = delegate.parseBeanDefinitionElement(
    ele); > 1
  if(bdHolder != null) {
    //当返回的 bdHolder 不为空的情况下,如果存在默认标签的子节点下
    //再有自定义标签, 需要再次对自定义标签进行解析
    bdHolder = delegate.decorateBeanDefinitionIfRequired(ele,
      bdHolder);
    try {
      // Register the final decorated instance.
      //注册解析得到的BeanDefinition
      BeanDefinitionReaderUtils.registerBeanDefinition(bdHolder,
        getReaderContext().getRegistry()); > 2
    }
    catch(BeanDefinitionStoreException ex) {
      getReaderContext().error(
        "Failed to register bean definition with name '" +
        bdHolder.getBeanName() + "'", ele, ex);
    }
    // Send registration event.
    //通知相关的监昕器,这个 bean 加载完成了
    getReaderContext().fireComponentRegistered(new BeanComponentDefinition(
      bdHolder));
  }
}

解析 BeanDefinition

首先进入delegate.parseBeanDefinitionElement(ele);函数

@Nullable
public BeanDefinitionHolder parseBeanDefinitionElement(Element ele, @
  Nullable BeanDefinition containingBean) {
  //解析id属性
  String id = ele.getAttribute(ID_ATTRIBUTE);
  //解析name属性
  String nameAttr = ele.getAttribute(NAME_ATTRIBUTE);
  List < String > aliases = new ArrayList < > ();
  // 将bean元素的name放入别名数组中,spring支持两种方式定义别名,
  // 一种是通过<alias>,另一种是<bean name=""/>
  if(StringUtils.hasLength(nameAttr)) {
    //分割nameAttr
    String[] nameArr = StringUtils.tokenizeToStringArray(nameAttr,
      MULTI_VALUE_ATTRIBUTE_DELIMITERS);
    aliases.addAll(Arrays.asList(nameArr));
  }
  String beanName = id;
  // 如果beanName为空 并且别名集合不为空,将第一个别名作为beanName
  if(!StringUtils.hasText(beanName) && !aliases.isEmpty()) {
    beanName = aliases.remove(0);
    if(logger.isTraceEnabled()) {
      logger.trace("No XML 'id' specified - using '" + beanName +
        "' as bean name and " + aliases + " as aliases");
    }
  }
  // 内部Bean为空时
  // 什么情况下不为空呢?也就是有内部类时
  /**
   * <bean  class="com.gongj.bean.User" name="user2,use" id="user">
   *         <property name="vip">
   *             <bean id="inner" class="com.gongj.bean.User$Vip">
   *                 <constructor-arg ref="user"/>
   *                 <property name="vipLevel" value="9"/>
   *             </bean>
   *         </property>
   *     </bean>
   *
   */
  if(containingBean == null) {
    //验证指定的bean名称(也就是id属性)和别名是否已经存在
    checkNameUniqueness(beanName, aliases, ele);
  }
  // 对bean标签的其他属性进行解析,并封装在 GenericBeanDefinition 类型的实例中 
  AbstractBeanDefinition beanDefinition =
    parseBeanDefinitionElement(ele, beanName, containingBean); > 1
  if(beanDefinition != null) {
    //如果不存在 beanName
    if(!StringUtils.hasText(beanName)) {
      try {
        if(containingBean != null) {
          //那么根据 BeanDefinitionReaderUtils 提供的命名规则为当前 
          //bean 生成对应的beanName
          beanName = BeanDefinitionReaderUtils.generateBeanName(
            beanDefinition, this.readerContext.getRegistry(),
            true);
        }
        else {
          // 根据 XmlReaderContext 生成对应的beanName
          //类似于 com.gongj.bean.User#0
          beanName = this.readerContext.generateBeanName(
            beanDefinition);
          // 类似于:com.gongj.bean.User
          String beanClassName = beanDefinition.getBeanClassName();
          if(beanClassName != null && beanName.startsWith(
              beanClassName) && beanName.length() > beanClassName.length() &&
            !this.readerContext.getRegistry().isBeanNameInUse(
              beanClassName)) {
            //放入别名集合中
            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);
    //将获取到的信息封装到 BeanDefinitionHolder 实例中
    return new BeanDefinitionHolder(beanDefinition, beanName,
      aliasesArray);
  }
  return null;
}

进入标记点 >1parseBeanDefinitionElement(ele, beanName, containingBean)函数,对标签其他属性的解析过程。

@Nullable
public AbstractBeanDefinition parseBeanDefinitionElement(Element ele,
  String beanName, @Nullable BeanDefinition containingBean) {
  this.parseState.push(new BeanEntry(beanName));
  String className = null;
  //class
  if(ele.hasAttribute(CLASS_ATTRIBUTE)) {
    className = ele.getAttribute(CLASS_ATTRIBUTE).trim();
  }
  String parent = null;
  //parent
  if(ele.hasAttribute(PARENT_ATTRIBUTE)) {
    parent = ele.getAttribute(PARENT_ATTRIBUTE);
  }
  try {
    //将class与parentName作为参数,创建一个 GenericBeanDefinition 实例
    AbstractBeanDefinition bd = createBeanDefinition(className,
      parent); > 1
    //解析默认 bean 标签的各种属性,通过set方法进行属性赋值
    parseBeanDefinitionAttributes(ele, beanName, containingBean, bd); >2
    //提取description
    bd.setDescription(DomUtils.getChildElementValueByTagName(ele,
      DESCRIPTION_ELEMENT));
    // 解析meta元数据
    parseMetaElements(ele, bd);
    //解析lookup-method属性
    parseLookupOverrideSubElements(ele, bd.getMethodOverrides());
    //解析replaced-method属性
    parseReplacedMethodSubElements(ele, bd.getMethodOverrides());
    //解析构造函数参数
    parseConstructorArgElements(ele, bd);
    //解析property子元素
    parsePropertyElements(ele, bd);
    // 解析qualifier子元素
    parseQualifierElements(ele, bd);
    bd.setResource(this.readerContext.getResource());
    bd.setSource(extractSource(ele));
    return bd;
  }
  catch(ClassNotFoundException ex) {
    error("Bean class [" + className + "] not found", ele, ex);
  }
  catch(NoClassDefFoundError err) {
    error("Class that bean class [" + className +
      "] depends on not found", ele, err);
  }
  catch(Throwable ex) {
    error("Unexpected failure during bean definition parsing", ele,
      ex);
  }
  finally {
    this.parseState.pop();
  }
  return null;
}

进入createBeanDefinition(className, parent)函数

protected AbstractBeanDefinition createBeanDefinition(@Nullable
String className, @Nullable String parentName) 
throws ClassNotFoundException {
  return BeanDefinitionReaderUtils.createBeanDefinition(parentName,
    className, this.readerContext.getBeanClassLoader());
}

然后再调试进入`createBeanDefinition(

            parentName, className, this.readerContext.getBeanClassLoader())`函数
public static AbstractBeanDefinition createBeanDefinition(@Nullable String parentName, 
@Nullable String className, @Nullable ClassLoader classLoader) throws ClassNotFoundException {
  GenericBeanDefinition bd = new GenericBeanDefinition();
  //parentName可能为空
  bd.setParentName(parentName);
  if(className != null) {
    if(classLoader != null) {
      //如果 classLoader 不为空, 则使用传入的 classLoader 加载类对象,否则只是
      //记录className
      bd.setBeanClass(ClassUtils.forName(className, classLoader));
    }
    else {
      bd.setBeanClassName(className);
    }
  }
  return bd;
}

联系之前博文:Spring源码(一)-Bean的定义-BeanDefinition,可以进行考证,如果是 xml 配置,会解析所有属性并统一封装至 GenericBeanDefinition 类型的实例中,之后再逐渐解析。

然后再看一下 标记点 2 那个函数 parseBeanDefinitionAttributes(ele, beanName, containingBean, bd); 解析默认 bean 标签的各种属性。

public AbstractBeanDefinition parseBeanDefinitionAttributes(Element ele,
  String beanName, @Nullable BeanDefinition containingBean,
  AbstractBeanDefinition bd) {
  //解析 singleton 属性
  if(ele.hasAttribute(SINGLETON_ATTRIBUTE)) {
    //'singleton'属性升级到'scope'声明
    error(
      "Old 1.x 'singleton' attribute in use - upgrade to 'scope' declaration",
      ele);
  }
  //解析 scope 属性
  else if(ele.hasAttribute(SCOPE_ATTRIBUTE)) {
    bd.setScope(ele.getAttribute(SCOPE_ATTRIBUTE));
  }
  //当当前Bean没有明确使用 scope 属性时,且当前Bean被解析为是内部bean的话,
  //则默认使用外部Bean的作用域作为当前bean的作用域
  else if(containingBean != null) {
    // Take default from containing bean in case of an inner bean definition.
    bd.setScope(containingBean.getScope());
  }
  //解析 abstract 属性
  if(ele.hasAttribute(ABSTRACT_ATTRIBUTE)) {
    bd.setAbstract(TRUE_VALUE.equals(ele.getAttribute(
      ABSTRACT_ATTRIBUTE)));
  }
  //解析 lazy-init 属性
  String lazyInit = ele.getAttribute(LAZY_INIT_ATTRIBUTE);
  if(isDefaultValue(lazyInit)) {
    lazyInit = this.defaults.getLazyInit();
  }
  // 若没有设置或设置为其他字符都会被设置为false
  bd.setLazyInit(TRUE_VALUE.equals(lazyInit));
  //解析 autowire 属性
  String autowire = ele.getAttribute(AUTOWIRE_ATTRIBUTE);
  bd.setAutowireMode(getAutowireMode(autowire));
  //解析 depends-on 属性
  if(ele.hasAttribute(DEPENDS_ON_ATTRIBUTE)) {
    String dependsOn = ele.getAttribute(DEPENDS_ON_ATTRIBUTE);
    bd.setDependsOn(StringUtils.tokenizeToStringArray(dependsOn,
      MULTI_VALUE_ATTRIBUTE_DELIMITERS));
  }
  //解析 autowire-candidate 属性
  String autowireCandidate = ele.getAttribute(
    AUTOWIRE_CANDIDATE_ATTRIBUTE);
  if(isDefaultValue(autowireCandidate)) {
    String candidatePattern = this.defaults.getAutowireCandidates();
    if(candidatePattern != null) {
      String[] patterns = StringUtils.commaDelimitedListToStringArray(
        candidatePattern);
      bd.setAutowireCandidate(PatternMatchUtils.simpleMatch(
        patterns, beanName));
    }
  }
  else {
    bd.setAutowireCandidate(TRUE_VALUE.equals(autowireCandidate));
  }
  //解析 primary 属性
  if(ele.hasAttribute(PRIMARY_ATTRIBUTE)) {
    bd.setPrimary(TRUE_VALUE.equals(ele.getAttribute(
      PRIMARY_ATTRIBUTE)));
  }
  //解析 init-method 属性
  if(ele.hasAttribute(INIT_METHOD_ATTRIBUTE)) {
    String initMethodName = ele.getAttribute(INIT_METHOD_ATTRIBUTE);
    bd.setInitMethodName(initMethodName);
  }
  else if(this.defaults.getInitMethod() != null) {
    bd.setInitMethodName(this.defaults.getInitMethod());
    bd.setEnforceInitMethod(false);
  }
  //解析 destroy-method 属性
  if(ele.hasAttribute(DESTROY_METHOD_ATTRIBUTE)) {
    String destroyMethodName = ele.getAttribute(
      DESTROY_METHOD_ATTRIBUTE);
    bd.setDestroyMethodName(destroyMethodName);
  }
  else if(this.defaults.getDestroyMethod() != null) {
    bd.setDestroyMethodName(this.defaults.getDestroyMethod());
    bd.setEnforceDestroyMethod(false);
  }
  //解析 factory-method 属性
  if(ele.hasAttribute(FACTORY_METHOD_ATTRIBUTE)) {
    bd.setFactoryMethodName(ele.getAttribute(
      FACTORY_METHOD_ATTRIBUTE));
  }
  //解析 factory-bean 属性
  if(ele.hasAttribute(FACTORY_BEAN_ATTRIBUTE)) {
    bd.setFactoryBeanName(ele.getAttribute(FACTORY_BEAN_ATTRIBUTE));
  }
  return bd;
}

每个属性的作用可以看笔者的之前源码:Spring源码(一)-Bean的定义-BeanDefinition

注册BeanDefinition

到这里Spring已经将 xml 中的配置信息封装至BeanDefinitionHolder对象,接下来就可以执行对 BeanDefinition 的注册了,一起继续向下吧。调试进入 `BeanDefinitionReaderUtils.registerBeanDefinition(bdHolder,
getReaderContext().getRegistry())`函数

public static void registerBeanDefinition(BeanDefinitionHolder definitionHolder,
  BeanDefinitionRegistry registry)
throws BeanDefinitionStoreException {
  // Register bean definition under primary name.
  // 注册bean定义
  String beanName = definitionHolder.getBeanName();
  registry.registerBeanDefinition(beanName, definitionHolder.getBeanDefinition()); >
  1
  // Register aliases for bean name, if any.
  // 注册bean名称的别名(如果有的话)
  String[] aliases = definitionHolder.getAliases();
  if(aliases != null) {
    for(String alias: aliases) {
      registry.registerAlias(beanName, alias); > 2
    }
  }
}

从上面的代码可以看出 ,解析的 BeanDefinition 都会被注册到 BeanDefinitionRegistry 类型的实例中, 对于 BeanDefinition 的注册分成了两部分:通过 beanName 的注册以及通过别名的注册。


通过beanName注册

首先进入registry.registerBeanDefinition(beanName,definitionHolder.getBeanDefinition())函数,进入该函数会进入到 BeanDefinitionRegistry接口,该接口下有三个子类,我们选择进入默认使用的DefaultListableBeanFactory子类

@Override
public void registerBeanDefinition(String beanName, BeanDefinition beanDefinition)
throws BeanDefinitionStoreException {
  Assert.hasText(beanName, "Bean name must not be empty");
  Assert.notNull(beanDefinition, "BeanDefinition must not be null");
  if(beanDefinition instanceof AbstractBeanDefinition) {
    try {
      /*
      *注册前的最后一次校验
      主要是对于 AbstractBeanDefinition 中的 methodOverrides 属性校验,
      * 校验 methodOverrides 是否与工厂方法并存或者 methodOverrides 
      对应的方法根本不存在
      */
      ((AbstractBeanDefinition) beanDefinition).validate();
    }
    catch(BeanDefinitionValidationException ex) {
      throw new BeanDefinitionStoreException(beanDefinition.getResourceDescription(),
        beanName, "Validation of bean definition failed", ex);
    }
  }
  // beanDefinitionMap: bean定义对象的映射 key为beanName
  // 根据beanName获取BeanDefinition
  BeanDefinition existingDefinition = this.beanDefinitionMap.get(
    beanName);
  //处理已经注册的 Bean 情况
  if(existingDefinition != null) {
    //如果对应的 Bean 已经注册且在配置中配置了 bean 不允许被覆盖,
    //则抛出异常,默认是 true,也就是允许覆盖
    if(!isAllowBeanDefinitionOverriding()) {
      throw new BeanDefinitionOverrideException(beanName,
        beanDefinition, existingDefinition);
    }
    // getRole方法可以去看上篇 BeanDefinition
    // 使用新的BeanDefinition覆盖已经加载的BeanDefinition
    else if(existingDefinition.getRole() < beanDefinition.getRole()) {
      if(logger.isInfoEnabled()) {
        logger.info(
          "Overriding user-defined bean definition for bean '" +
          beanName +
          "' with a framework-generated bean definition: replacing [" +
          existingDefinition + "] with [" + beanDefinition + "]");
      }
    }
    else if(!beanDefinition.equals(existingDefinition)) {
      if(logger.isDebugEnabled()) {
        logger.debug("Overriding bean definition for bean '" +
          beanName + "' with a different definition: replacing [" +
          existingDefinition + "] with [" + beanDefinition + "]");
      }
    }
    else {
      if(logger.isTraceEnabled()) {
        logger.trace("Overriding bean definition for bean '" +
          beanName + "' with an equivalent definition: replacing [" +
          existingDefinition + "] with [" + beanDefinition + "]");
      }
    }
    //注册 BeanDefinition,加入 beanDefinitionMap 缓存。
    this.beanDefinitionMap.put(beanName, beanDefinition);
  }
  //处理未注册的情况
  else {
    // 如果beanDefinition已经被标记为创建(为了解决单例bean的循环依赖问题)
    if(hasBeanCreationStarted()) {
      
      synchronized(this.beanDefinitionMap) {
        //注册 BeanDefinition
        this.beanDefinitionMap.put(beanName, beanDefinition);
        // 创建List<String>并将缓存的beanDefinitionNames和新的beanName加入集合
        List < String > updatedDefinitions = new ArrayList < > (
          this.beanDefinitionNames.size() + 1);
        updatedDefinitions.addAll(this.beanDefinitionNames);
        updatedDefinitions.add(beanName);
        this.beanDefinitionNames = updatedDefinitions;
        removeManualSingletonName(beanName); > 1
      }
    }
    else {
      // Still in startup registration phase
      // 仍处于启动注册阶段
      this.beanDefinitionMap.put(beanName, beanDefinition);
      //维护了beanName集合
      this.beanDefinitionNames.add(beanName);
      removeManualSingletonName(beanName); > 1
    }
    this.frozenBeanDefinitionNames = null;
  }
  // 当前注册的bean的定义已经在beanDefinitionMap缓存中存在,
  // 或者其实例已经存在于单例bean的缓存中(singletonObjects),
  //就进行重置当前Bean的BeanDefinition
  if(existingDefinition != null || containsSingleton(beanName)) {
    resetBeanDefinition(beanName);
  }
}
protected boolean hasBeanCreationStarted() {
  return !this.alreadyCreated.isEmpty();
}

我们看一下removeManualSingletonName(beanName)这个函数,这个函数使用了 java8 的函数式接口。

private void removeManualSingletonName(String beanName) {
  updateManualSingletonNames(set - > set.remove(beanName), set - >
    set.contains(beanName));
}
private void updateManualSingletonNames(Consumer < Set < String >>
  action, Predicate < Set < String >> condition) {
  if(hasBeanCreationStarted()) {
   
    synchronized(this.beanDefinitionMap) {
      //如果manualSingletonNames中包含新注册的beanName
      //对应 set -> set.contains(beanName)
      if(condition.test(this.manualSingletonNames)) {
        // 创建set集合并将manualSingletonNames加入到新创建的set集合
        Set < String > updatedSingletons = new LinkedHashSet < >
          (this.manualSingletonNames);
        //然后移除新注册的beanName
        //对应set.remove(beanName)
        action.accept(updatedSingletons);
        this.manualSingletonNames = updatedSingletons;
      }
    }
  }
  else {
    // Still in startup registration phase
    if(condition.test(this.manualSingletonNames)) {
      action.accept(this.manualSingletonNames);
    }
  }
}

第一个函数表达式入参为一个 Consumer 消费型接口,代表接受一个输入参数并且无返回的操作
第二个函数表达式入参为一个 Predicate 断言形接口,接受一个输入参数,返回一个布尔值结果。


接下来就看看上面代码所用的数据存储容器:

private final Map<String, BeanDefinition> beanDefinitionMap = new ConcurrentHashMap<>(256);
private volatile List<String> beanDefinitionNames = new ArrayList<>(256);
private final Set<String> alreadyCreated = Collections.newSetFromMap(new ConcurrentHashMap<>(256));
private volatile Set<String> manualSingletonNames = new LinkedHashSet<>(16);
  • beanDefinitionMap:bean定义对象的映射,key为beanName,value为BeanDefinition
  • beanDefinitionNames :beanName列表
  • alreadyCreated :存储至少创建了一次的bean的名称
  • manualSingletonNames:手动注册的单例的名称列表

通过别名注册

进入registry.registerAlias(beanName, alias)函数,会跳转到 AliasRegistry接口,该接口下有两个实现类 GenericApplicationContextSimpleAliasRegistry,我们选择SimpleAliasRegistry

@Override
public void registerAlias(String name, String alias) {
  Assert.hasText(name, "'name' must not be empty");
  Assert.hasText(alias, "'alias' must not be empty");
  synchronized(this.aliasMap) {
    //别名与beanName相同
    if(alias.equals(name)) {
      //将该别名从Map中移除
      this.aliasMap.remove(alias);
      if(logger.isDebugEnabled()) {
        logger.debug("Alias definition '" + alias +
          "' ignored since it points to same name");
      }
    }
    //别名与beanName不相同
    else {
      // 根据传入的别名获取beanName
      String registeredName = this.aliasMap.get(alias);
      if(registeredName != null) {
        //已经注册的名称与当前需要注册的beanName相等
        /*
        <alias name="user" alias="user2"></alias>
        <alias name="user" alias="user2"></alias>
         */
        if(registeredName.equals(name)) {
          // An existing alias - no need to re-register
          //已存在的别名-不需要重新注册
          return;
        }
        //如果 alias 不允许被覆盖则抛出异常,默认是true,允许被覆盖
        if(!allowAliasOverriding()) {
          throw new IllegalStateException("Cannot define alias '" +
            alias + "' for name '" + name +
            "': It is already registered for name '" +
            registeredName + "'.");
        }
        if(logger.isDebugEnabled()) {
          logger.debug("Overriding alias '" + alias +
            "' definition for registered name '" +
            registeredName + "' with new target name '" + name +
            "'");
        }
      }
      // 检查给定的名称(user2)是否指向给定的别名(user)作为别名(user2)
      /*
         2、<alias name="user" alias="user2"></alias>
         1、<alias name="user2" alias="user"></alias>
          */
      checkForAliasCircle(name, alias);
      //缓存别名
      this.aliasMap.put(alias, name);
      if(logger.isTraceEnabled()) {
        logger.trace("Alias definition '" + alias +
          "' registered for name '" + name + "'");
      }
    }
  }
}

bean 标签的解析及注册解析完成啦!

相关文章
|
3天前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
15 2
|
19天前
|
数据采集 监控 前端开发
二级公立医院绩效考核系统源码,B/S架构,前后端分别基于Spring Boot和Avue框架
医院绩效管理系统通过与HIS系统的无缝对接,实现数据网络化采集、评价结果透明化管理及奖金分配自动化生成。系统涵盖科室和个人绩效考核、医疗质量考核、数据采集、绩效工资核算、收支核算、工作量统计、单项奖惩等功能,提升绩效评估的全面性、准确性和公正性。技术栈采用B/S架构,前后端分别基于Spring Boot和Avue框架。
|
10天前
|
缓存 Java Spring
实战指南:四种调整 Spring Bean 初始化顺序的方案
本文探讨了如何调整 Spring Boot 中 Bean 的初始化顺序,以满足业务需求。文章通过四种方案进行了详细分析: 1. **方案一 (@Order)**:通过 `@Order` 注解设置 Bean 的初始化顺序,但发现 `@PostConstruct` 会影响顺序。 2. **方案二 (SmartInitializingSingleton)**:在所有单例 Bean 初始化后执行额外的初始化工作,但无法精确控制特定 Bean 的顺序。 3. **方案三 (@DependsOn)**:通过 `@DependsOn` 注解指定 Bean 之间的依赖关系,成功实现顺序控制,但耦合性较高。
实战指南:四种调整 Spring Bean 初始化顺序的方案
|
8天前
|
Java Maven
maven项目的pom.xml文件常用标签使用介绍
第四届人文,智慧教育与服务管理国际学术会议(HWESM 2025) 2025 4th International Conference on Humanities, Wisdom Education and Service Management
55 8
|
9天前
|
前端开发 Java 开发者
Spring生态学习路径与源码深度探讨
【11月更文挑战第13天】Spring框架作为Java企业级开发中的核心框架,其丰富的生态系统和强大的功能吸引了无数开发者的关注。学习Spring生态不仅仅是掌握Spring Framework本身,更需要深入理解其周边组件和工具,以及源码的底层实现逻辑。本文将从Spring生态的学习路径入手,详细探讨如何系统地学习Spring,并深入解析各个重点的底层实现逻辑。
35 9
|
28天前
|
Java 测试技术 Windows
咦!Spring容器里为什么没有我需要的Bean?
【10月更文挑战第11天】项目经理给小菜分配了一个紧急需求,小菜迅速搭建了一个SpringBoot项目并完成了开发。然而,启动测试时发现接口404,原因是控制器包不在默认扫描路径下。通过配置`@ComponentScan`的`basePackages`字段,解决了问题。总结:`@SpringBootApplication`默认只扫描当前包下的组件,需要扫描其他包时需配置`@ComponentScan`。
|
1月前
|
Java Spring
Spring底层架构源码解析(三)
Spring底层架构源码解析(三)
108 5
|
1月前
|
XML Java 数据格式
Spring底层架构源码解析(二)
Spring底层架构源码解析(二)
|
1月前
|
XML Java 数据格式
手动开发-简单的Spring基于注解配置的程序--源码解析
手动开发-简单的Spring基于注解配置的程序--源码解析
46 0
|
1月前
|
XML Java 数据格式
手动开发-简单的Spring基于XML配置的程序--源码解析
手动开发-简单的Spring基于XML配置的程序--源码解析
79 0