当前位置: 首页 > wzjs >正文

桥头做网站预约网页怎么制作

桥头做网站,预约网页怎么制作,商城网站多少钱,wordpress下载类模板Spring Value注解的依赖注入实现原理 一,什么是Value注解的依赖注入二,实现原理三,代码实现1. 定义 Value 注解2. 实现 InstantiationAwareBeanPostProcessor3. 实现 AutowiredAnnotationBeanPostProcessor4. 占位符解析逻辑5. 定义 StringVa…

Spring @Value注解的依赖注入实现原理

    • 一,什么是Value注解的依赖注入
    • 二,实现原理
    • 三,代码实现
      • 1. 定义 @Value 注解
      • 2. 实现 InstantiationAwareBeanPostProcessor
      • 3. 实现 AutowiredAnnotationBeanPostProcessor
      • 4. 占位符解析逻辑
      • 5. 定义 StringValueResolver 接口
      • 6. 实现 PlaceholderResolvingStringValueResolver
      • 7. 注册解析器
      • 8. 集成到 BeanFactory

源码见:mini-spring

在这里插入图片描述

一,什么是Value注解的依赖注入

在 Spring 框架中,@Value 注解是一种常用的依赖注入方式,用于直接为 Bean 的属性注入值。以下是一个简单的示例:

@Value("jixu")
private String name;

除了直接注入静态值,@Value 还支持属性占位符,能够从配置文件(如 application.propertiesapplication.yml)中动态读取值。例如:

@Value("${sex}")
private String sex;

通过这种方式,我们可以灵活地将外部配置的值注入到 Bean 的属性中。

二,实现原理

要理解 @Value 注解的工作原理,首先需要明确其作用的时机和位置。显然,@Value 的功能需要在 Bean 属性赋值操作之前完成。具体来说,我们需要修改 BeanDefinition,为其添加对应的 PropertyValue,从而确保在 Bean 实例化并执行属性赋值时,能够通过 set 方法正确注入值。

位置已经确定了,那么再具体一点需要依赖的组建也就是BeanPostProcess,再精确一点是InstantiationAwareBeanPostProcessor,也就是我们在实现AOP融入Bean生命周期的时候定义的接口,该接口是用于处理实例化的相关操作,因此该扩展功能也会在这里实现。我们通之前的操作逻辑一样会定义一个抽象方法用于属性赋值,之后在我们具体的实现类当中实现相关逻辑,再加入 到对应的BeanPostProcess容器当中,之后在AbstractAutowireCapableBeanFactory的对应位置进行处理即可。

  1. 定义抽象方法:与 Spring 中其他扩展逻辑类似,我们可以先定义一个抽象方法,用于处理属性的赋值逻辑。
  2. 实现具体逻辑:在具体的实现类中,解析 @Value 注解,提取注解中的值(或占位符),并将其转换为 PropertyValue,附加到 BeanDefinition 上。
  3. 注册到容器:将实现的 InstantiationAwareBeanPostProcessor 加入到 Spring 的 BeanPostProcessor 容器中。
  4. 集成到生命周期:Spring 会在 AbstractAutowireCapableBeanFactory 的适当位置调用我们的处理器,完成 @Value 注解的处理。

三,代码实现

1. 定义 @Value 注解

首先,定义自定义的 @Value 注解:

@Target({ElementType.FIELD,ElementType.METHOD,ElementType.PARAMETER})  
@Retention(RetentionPolicy.RUNTIME)  
@Documented  
public @interface Value {  String value();  
}

2. 实现 InstantiationAwareBeanPostProcessor

在 InstantiationAwareBeanPostProcessor 中添加用于处理 PropertyValues 的方法:

PropertyValues postProcessPropertyValues(PropertyValues propertyValues , Object bean , String beanName);

3. 实现 AutowiredAnnotationBeanPostProcessor

定义 AutowiredAnnotationBeanPostProcessor 类,实现 InstantiationAwareBeanPostProcessor 接口,并在 postProcessPropertyValues 方法中实现具体的解析逻辑:

  
@Component  
public class AutowiredAnnotationBeanPostProcessor implements InstantiationAwareBeanPostProcessor , BeanFactoryAware {  private ConfigurableBeanFactory beanFactory;  @Override  public PropertyValues postProcessPropertyValues(PropertyValues propertyValues, Object bean, String beanName) {  Class<?> beanClass = bean.getClass();  PropertyValues pvs = new PropertyValues();  // 获取到当前类当中声明的所有属性  Field[] declaredFields = beanClass.getDeclaredFields();  for (Field field : declaredFields) {  // 获取到标记Value注解的属性  Value valueAnnotation = field.getAnnotation(Value.class);  if (valueAnnotation != null){  String value = valueAnnotation.value();  // 解析Value的属性值,判断是否需要替换占位符  value = beanFactory.resolveEmbeddedValue(value);  // 将解析完毕的字段添加到类属性当中  // BeanUtil.setFieldValue(bean,field.getName(),value);  pvs.addPropertyValue(new PropertyValue(field.getName(), value));  }  }  return pvs;  }  @Override  public Object postProcessBeforeInstantiation(Class<?> beanClass, String beanName) {  return null;  }  /**  * 在 Bean 初始化之前执行自定义处理逻辑。  * 使用此方法,可以在 Bean 被初始化之前对其进行修改或执行其他操作。  *  * @param bean     当前正在初始化的 Bean 实例。  * @param beanName 当前 Bean 的名称。  * @return 返回处理后的 Bean 实例,可以是原始 Bean 或修改后的 Bean。  */  @Override  public Object postProcessBeforeInitialization(Object bean, String beanName) {  return null;  }  /**  * 在 Bean 初始化之后执行自定义处理逻辑。  * 使用此方法,可以在 Bean 初始化完成后对其进行进一步的修改或执行其他操作。  *  * @param bean     当前已经初始化的 Bean 实例。  * @param beanName 当前 Bean 的名称。  * @return 返回处理后的 Bean 实例,可以是原始 Bean 或修改后的 Bean。  */  @Override  public Object postProcessAfterInitialization(Object bean, String beanName) {  return null;  }  @Override  public void setBeanFactory(BeanFactory beanFactory) {  this.beanFactory = (ConfigurableBeanFactory) beanFactory;  }  
}

说明:postProcessPropertyValues 方法接收的 propertyValues 参数包含当前 Bean 的所有属性值,但在此处未充分利用(例如未检查属性重复)。方法通过反射获取字段,检查 @Value 注解,解析值后添加到一个新的 PropertyValues 对象中返回。相比官方实现(使用 MutablePropertyValues 合并原始 propertyValues),此实现进行了简化。

4. 占位符解析逻辑

在上述逻辑中,resolveEmbeddedValue 方法用于解析占位符。定义如下:

// 在 ConfigurableBeanFactory 中定义
String resolveEmbeddedValue(String value);/*** 添加属性解析器,以便解析嵌入值中的占位符* @param stringValueResolver 属性解析器*/
void addEmbeddedValueResolver(StringValueResolver stringValueResolver);

由 AbstractBeanFactory 实现:

/**  * 解析嵌入值,用于Value注解解析  *  * @param value * @return */@Override  
public String resolveEmbeddedValue(String value) {  String result = value;  for (StringValueResolver resolver : embeddedValueResolvers) {  // 会判断传入字段是否包含属性占位符,如果包含则替换为配置文件当中的值  result = resolver.resolveStringValue(result);  }  return result;  }  /**  * 添加属性解析器  *  * @param stringValueResolver */@Override  
public void addEmbeddedValueResolver(StringValueResolver stringValueResolver) {  embeddedValueResolvers.add(stringValueResolver);  
}

疑问解答:addEmbeddedValueResolver 的作用在于支持多个配置文件。Spring 可能配置多个 PropertyPlaceholderConfigurer,每个对应一个配置文件。通过添加多个 StringValueResolver,可以依次解析占位符,确保所有配置文件的属性值都被正确替换。

5. 定义 StringValueResolver 接口

定义工具类接口以抽象占位符解析逻辑:

public interface StringValueResolver {  String resolveStringValue(String strVal);  }

6. 实现 PlaceholderResolvingStringValueResolver

在 PropertyPlaceholderConfigurer 中定义内部类实现该接口:

定义一个内部类PlaceholderResolvingStringValueResolver实现该工具类的方法,这样就为所有的PropertyPlaceholderConfigurer提供了一层抽象层的实现,用于解析占位符。

/**  * 定义字符解析器  */  
public class PlaceholderResolvingStringValueResolver implements StringValueResolver {  // 配置文件Properties对象  private final Properties properties;  public PlaceholderResolvingStringValueResolver(Properties properties) {  this.properties = properties;  }  @Override  public String resolveStringValue(String strVal) {  return PropertyPlaceholderConfigurer.this.resolverPlaceholder(strVal,properties);  }  
}

7. 注册解析器

在 PropertyPlaceholderConfigurer 的 postProcessBeanFactory 方法中注册解析器:

public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) {  // 加载属性配置文件  Properties properties = loadProperties();  // 属性值替换占位符  processProperties(beanFactory, properties);  // 添加属性解析器  StringValueResolver resolver = new PlaceholderResolvingStringValueResolver(properties);  beanFactory.addEmbeddedValueResolver(resolver);  
}

说明:postProcessBeanFactory 在容器刷新时执行,扫描所有 BeanFactoryPostProcessor 实现(如 PropertyPlaceholderConfigurer)。每个 PropertyPlaceholderConfigurer 对应一个配置文件,确保多配置文件场景下的占位符解析。

8. 集成到 BeanFactory

在 AbstractAutowireCapableBeanFactory 的 doCreateBean 方法中,实例化后、赋值前执行:

// 通过InstantiationStrategy实例化Bean  
bean = createBeanInstance(beanDefinition);  applyBeanPostprocessorsBeforeApplyingPropertyValues(beanName, bean, beanDefinition);  // 为Bean的属性进行赋值  
applyPropertyValues(bean , beanDefinition , beanName);

具体实现逻辑如下:

private void applyBeanPostprocessorsBeforeApplyingPropertyValues(String beanName, Object bean, BeanDefinition beanDefinition) {  List<BeanPostProcessor> beanPostProcessors = getBeanPostProcessors();  for (BeanPostProcessor beanPostProcessor : beanPostProcessors) {  if (beanPostProcessor instanceof  InstantiationAwareBeanPostProcessor) {  PropertyValues propertyValues = ((InstantiationAwareBeanPostProcessor) beanPostProcessor).postProcessPropertyValues(beanDefinition.getPropertyValues(), bean, beanName);  if (propertyValues != null) {  for (PropertyValue propertyValue : propertyValues.getPropertyValueList()) {  beanDefinition.getPropertyValues().addPropertyValue(propertyValue);  }  }  }  }  
}

说明:该逻辑与 AOP 融入生命周期的实现类似,通过循环调用 BeanPostProcessor,处理并合并 PropertyValues。


文章转载自:

http://deD3JNQ8.nbwyk.cn
http://lmQeSGgI.nbwyk.cn
http://HMAEQc7Z.nbwyk.cn
http://X6Bl46TX.nbwyk.cn
http://nZPsuZaJ.nbwyk.cn
http://DWKlGGfK.nbwyk.cn
http://5CobJGv8.nbwyk.cn
http://nIN8hz3g.nbwyk.cn
http://wVBAkzA6.nbwyk.cn
http://TXrBdFLF.nbwyk.cn
http://19RhWSIx.nbwyk.cn
http://wVx5W8eq.nbwyk.cn
http://goWJJvIA.nbwyk.cn
http://uDT51uE2.nbwyk.cn
http://laxFSNTA.nbwyk.cn
http://jYtOluFw.nbwyk.cn
http://YrkSCRKg.nbwyk.cn
http://lE1C5Qk1.nbwyk.cn
http://NGDXsqQR.nbwyk.cn
http://pa7SzgoI.nbwyk.cn
http://oN17iae5.nbwyk.cn
http://SiJIwUpq.nbwyk.cn
http://QdoxJuYP.nbwyk.cn
http://2otMMyNT.nbwyk.cn
http://D7cSlCKh.nbwyk.cn
http://jZGEgTpb.nbwyk.cn
http://y5LtcDP0.nbwyk.cn
http://MACvYICt.nbwyk.cn
http://yw64PPDg.nbwyk.cn
http://1JvOtTBp.nbwyk.cn
http://www.dtcms.com/wzjs/670819.html

相关文章:

  • 引物在线设计网站wordpress如何修改网站域名
  • 网站域名注册步骤大连网站建设怎么样
  • 上海松江做网站公司用seo对网站做分析
  • 站长工具最近查询中企动力科技股份
  • 北京网站备案拍照地址鼓楼网站开发与设计培训
  • 任丘网站制作中国建筑网官网查证
  • 长春网站建设网诚传媒做网站需要公司资料吗
  • 芜湖市网站建设一个网站做多少关键词
  • 做网站和服务器的大小有关吗wordpress istax
  • 上海网站建设设计百度服务中心官网
  • 软件介绍网站源码网站开发api和微端
  • 注册域名后网站建设留学网站建设文案
  • 图文网站模版html5 网站 优势
  • 怎么建设网站是什么建网站 西安
  • 运城网站建设多少钱专注旅游网站网站开发
  • 外贸建站服务微商城运营的主要工作
  • 哪些网站可以做邀请函西安制作手机网站
  • 平顶山专业做网站公司网站建设中怎么解决
  • 电影资源网站建设长春手机建站模板
  • 网站建设 海南南联做网站
  • 建设学校网站前的需求分析报告opencart zencart网站建设
  • 网站开发就业方向陕西网页制作公司
  • 怎么做网站生意云服务器里面做网站播放器
  • 网站开发者工作描述wordpress不显示引用图片不显示
  • 网站怎么适配移动端php按步骤做网站
  • dw自己做网站需要什么区别wordpress适合下载站的主题
  • 诸暨市住房和城乡建设局网站高级网络规划师报考条件
  • 广州公司注册网站官网管理系统定制开发流程
  • 成品网站1688入门网免费网站模板怎么做网站
  • 云服务器 能用来做网站吗一家装修的网站怎么做的