【Struts2源码】@Inject查找并缓存注入器

本文详细介绍了Struts框架中@Inject注解的工作原理。主要分析了ContainerImpl类如何通过@Inject注解进行依赖注入,包括构造函数、属性、方法及方法参数等不同场景的注入过程。同时探讨了注入器的查找与缓存机制。

注解@Inject的操作类是ContainerImpl,@Inject在Struts中用到了很多,主要用来构造函数、属性、方法及方法参数注入,当调用ContainerImpl的Inject方法时,这些部分就启动,ContainerImpl初始化时,缓存这些构建方法

这里我们只分析查找这些带@Inject的注入器,并把它们缓存起来。

1.初始化一个注入器Map,用来放置找到@Inject后实例化的注入器,这里使用ReferenceCache缓存类,在运行期构建map,核心方法create会调用addInjectors方法扫描所有注入器加入到ReferenceCache缓存Map里面,并且内部实现ThreadLocal模式规避多线程问题,

	/**
	 * Field and method injectors.
	 */
	final Map<Class<?>, List<Injector>> injectors =
			new ReferenceCache<Class<?>, List<Injector>>() {
				@Override
				protected List<Injector> create( Class<?> key ) {
					List<Injector> injectors = new ArrayList<Injector>();
					addInjectors(key, injectors);
					return injectors;
				}
			};

2.addInjectors()方法

	void addInjectors( Class clazz, List<Injector> injectors ) {
		if (clazz == Object.class) {
			return;
		}

		// Add injectors for superclass first.
		addInjectors(clazz.getSuperclass(), injectors);

		// TODO (crazybob): Filter out overridden members.
		addInjectorsForFields(clazz.getDeclaredFields(), false, injectors);
		addInjectorsForMethods(clazz.getDeclaredMethods(), false, injectors);
	}
	void addInjectorsForMethods( Method[] methods, boolean statics,
								 List<Injector> injectors ) {
		addInjectorsForMembers(Arrays.asList(methods), statics, injectors,
				new InjectorFactory<Method>() {
					public Injector create( ContainerImpl container, Method method,
											String name ) throws MissingDependencyException {
						return new MethodInjector(container, method, name);
					}
				});
	}

	void addInjectorsForFields( Field[] fields, boolean statics,
								List<Injector> injectors ) {
		addInjectorsForMembers(Arrays.asList(fields), statics, injectors,
				new InjectorFactory<Field>() {
					public Injector create( ContainerImpl container, Field field,
											String name ) throws MissingDependencyException {
						return new FieldInjector(container, field, name);
					}
				});
	}
3.找到有注解@Inject的成员,调用传入的injectorFactory.create方法创建真正的Injector实例 加入到注入器集合中

	<M extends Member & AnnotatedElement> void addInjectorsForMembers(
			List<M> members, boolean statics, List<Injector> injectors,
			InjectorFactory<M> injectorFactory ) {
		for ( M member : members ) {
			if (isStatic(member) == statics) {
				Inject inject = member.getAnnotation(Inject.class);
				if (inject != null) {
					try {
						injectors.add(injectorFactory.create(this, member, inject.value()));
					} catch ( MissingDependencyException e ) {
						if (inject.required()) {
							throw new DependencyException(e);
						}
					}
				}
			}
		}
	}
这样就可以无差别的在构造函数、属性、方法及方法参数处使用@Inject注解了



使用 @Inject 注解进行属性注入时,有以下注意事项: ### 依赖的存在性 要保证被注入的依赖在 Spring 容器中已经被正确定义为 Bean。若依赖不存在,在应用启动时就会抛出 `NoSuchBeanDefinitionException` 异常。例如下面代码,如果 `MyService` 没有在 Spring 容器中定义为 Bean,就会出现问题: ```java import javax.inject.Inject; class MyClient { @Inject private MyService myService; } ``` ### 依赖的唯一性 若容器里存在多个类型相同的 Bean,使用 @Inject 直接注入会引发 `NoUniqueBeanDefinitionException` 异常。此时可以结合 `@Named` 注解来明确指定要注入的 Bean。如下示例,容器中有两个 `DataSource` 类型的 Bean,用 `@Named` 注解指定注入的 Bean: ```java import javax.inject.Inject; import javax.inject.Named; class MyDatabaseService { @Inject @Named("primary") private DataSource dataSource; } ``` ### 属性不可为 final 由于属性注入是在对象创建之后进行的,所以使用 @Inject 注解的属性不能被声明为 `final`。因为 `final` 修饰的属性必须在对象创建时就进行初始化。示例如下,这样的代码是错误的: ```java import javax.inject.Inject; class MyClass { @Inject private final MyService myService; // 错误,不能为 final } ``` ### 循环依赖问题 使用 @Inject 进行属性注入时,可能会出现循环依赖的情况,也就是两个或多个 Bean 之间相互依赖。Spring 虽然对某些循环依赖情况有处理机制,但属性注入的循环依赖可能会导致 Bean 创建失败。可以通过构造器注入或者重新设计 Bean 之间的依赖关系来解决这个问题。 ### 跨框架兼容性 @Inject 是 JSR - 330 标准的注解,旨在提升跨框架的兼容性。不过不同的框架对该注解的支持可能存在差异,在多框架混合使用时,要确保各框架对 @Inject 注解的支持是一致的。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值