双向注入Dependency injection(依赖注入)和inversion of control(控制反转)现在对大多数Java 开发者来说都是熟悉的概念了。依赖注入允许一个组件通过容器“注入”另一个组件到一个setter方法或者实例变量的方式,来获得被“注入”组件的引用(reference)。我们之前看过的所有依赖注入的实现,注入发生在组件创建的时候,在此后实例的整个生命周期中不再改变。对无状态组件,这么做是有道理的。从客户端的角度来看,特定种类的无状态组件的所有实例都是可以替换的。另一方面,Seam着重处理有状态组件。此时传统的依赖注入不再是非常有效了。Seam引入了bijection(双向注入)这个名词,用来作为注入的广义概括。和injection(单向注入)对比,bijection是:
contextual(上下文相关的)- 双向注入用来针对不同的上下文来组装有状态组件(在较大范围的上下文中的组件,可以引用较小范围上下文中的组件)
bidirectional(双向的)- 被触发后,值从上下文变量中注射到组件属性中,也可以从组件属性outjected(反向注入)回上下文,这样被调用的组件可以只通过改写自己的实例变量就同时操作了上下文变量的值
dynamic(动态的)- 因为上下文变量的值随着时间不断改变,而且因为Seam组件是有状态的,双向注入在每次组件被调用的时候都发生。
基本上,通过设置实例变量是需要注入、反向注入、还是二者皆是,双向注入让你将上下文变量映射到组件实例变量。当然,我们使用注解来设置双向注入。
@In 注解指明应该注入值,可能是注入实例变量:
@Name("loginAction")@Statelesspublic class LoginAction implements Login { @In User user; ...}或者注入setter方法:
@Name("loginAction")@Statelesspublic class LoginAction implements Login { User user; @In public void setUser(User user) { this.user=user; } ...}默认情况下,针对被注入的属性或者实例变量名, Seam会对所有的上下文进行优先级搜索。 如果你希望明确指定上下文变量名,可以这样写:@In("currentUser")。
如果没有组件实例绑定到具名的上下文变量,你可能希望Seam创建一个,你可以指定 @In(create=true)。 如果值是可选的(可以为null),请指定 @In(required=false)。
对于某些组件,到处指定 @In(create=true) 是很繁琐的。 你可以注解整个组件为 @AutoCreate,它就会在任何需要的时候自动创建,不需要明确的指定 create=true。
你还可以注入表达式的值:
@Name("loginAction")@Statelesspublic class LoginAction implements Login { @In("#{user.username}") String username; ...}(在下一章,有更多的关于组件生命周期和注射的内容。)
@Out 注解指定了某个属性需要对外注入,可能是从实例变量:
@Name("loginAction")@Statelesspublic class LoginAction implements Login { @Out User user; ...}或者从某个getter方法:
@Name("loginAction")@Stateless@Interceptors(SeamInterceptor.class)public class LoginAction implements Login { User user; @Out public User getUser() { return user; } ...}属性可以既是被注入的,也可以对外注入:
@Name("loginAction")@Statelesspublic class LoginAction implements Login { @In @Out User user; ...}或者:
@Name("loginAction")@Statelesspublic class LoginAction implements Login { User user; @In public void setUser(User user) { this.user=user; } @Out public User getUser() { return user; } ...}