本文是<<struts2 技术内幕>>的学习笔记
在进行面向对象编程的时候,我们不可避免地要使用继承实现等等java提供的语法支持。但是复杂的对象关系也为对象生命周期的管理带来了最少以下两个问题。
1 程序运行时,应如何双肩我们所需要的对象。
2 当创建1个对象后,如何保证与其相干联的依赖关系也正确的被创建处理。
好在先辈们已给我们想好了前途------在程序中引入1个额外的编程元素:容器(Container)
对象的生命管理周期
首先我们得引入1个概念-----控制反转(Inverse of Control)
甚么是控制反转?从字面上来讲大概就是
本来应当由我控制的事情,不再需要我来控制了。至于交给谁来控制,我不需要知道。
举个例子。
public class Person{
private Car car;
public Person(Car c){
this.car=c;
}
public void drive(){
car.drive();
}
}
上面的例子很简单吧,1个人能开车。对车的产生,需要person自己来创造。
那甚么是ioc呢?
public class Person{
private Car car;
public Person(){
//其他代码
}
public void setCar(Car c){
this.car=c;
}
public void drive(){
car.drive();
}
}
就这么简单
本来需要person控制的车的创建,现在不由person来控制了。就是控制反转。
XWork容器的定义
刚才1直再说容器,现在我们就说说strtus2中的容器。
public interface Container extends Serializable {
/**
* Default dependency name.
*/
String DEFAULT_NAME = "default";
/**
* 创建类的实例,并进行注入 在上面说的person与car的例子中
* injcet的参数就应当是person
* object内部声明有@inject的字段和方法都将被注
* 入容器所托管的对象(例如上面说的car)
*/
void inject(Object o);
<T> T inject(Class<T> implementation);
/**
* 根据type与name取得容器中的java类实例
*/
<T> T getInstance(Class<T> type, String name);
/**
* 根据type和默许的name(default)取得容器中的java类实例
*/
<T> T getInstance(Class<T> type);
/**
* 根据type取得所有注册这个type的name
*/
Set<String> getInstanceNames(Class<?> type);
/**
* Sets the scope strategy for the current thread.
*/
void setScopeStrategy(Scope.Strategy scopeStrategy);
/**
* Removes the scope strategy for the current thread.
*/
void removeScopeStrategy();
}
接口中的方法可以分为3类
获得对象实例 getInstance,getInstacneNames
处理对象依赖关系 inject
处理对象的作用范围策略 setScopeStrategy,removeScopeStrategy
并且,容器是1个辅助的编程元素,它在系统中被设计为1个全局的单例的对象。
XWork容器的管辖范围
刚才我们已初步认识了XWork容器,现在我们看看:XWork到底能管理哪些东西。
1 获得对象实例
getInstance可以取得被容器托管的对象。那末到底哪些对象是容器所托管的呢?
看struts-default.xml里面有3类元素,是容器所托管的
1在bean节点中声明的框架内部的对象
2在bean节点中声明的自定义对象
3在constant节点中声明的系统运行参数
另外还有
在Properties文件里甚么歌的系统运行参数
再换句话说,想把自定义的对象纳入容器的管理范围,只需在Sturs/XWork的配置文件里声明便可。
2 对象的依赖注入
就像我们上面说的people与car的关系,people里面需要1个car,而car是通过容器所管理的,那末对people的注入就是container.inject(people)。
inject方法的参数都可使甚么呢?
调用XWork容器的inject方法,能够帮助我们将容器所管理的对象(包括框架的内部元素及系统运行参数)注入到任意的对象实例中。从而建立起任意对象与框架元素沟通的桥梁。
那末系统怎样知道people这个对象里面就需要1个car呢?
public class Person{
@Inject
private Car car;
public Person(){
//其他代码
}
@Inject
public void setCar(Car c){
this.car=c;
}
public void drive(){
car.drive();
}
}
在person里面的car参数或setCar方法上加上 @Inject标签
我们看看Inject标签的定义
@Target({METHOD, CONSTRUCTOR, FIELD, PARAMETER})
@Retention(RUNTIME)
public @interface Inject {
/**
* Dependency name. Defaults to {@link Container。DEFAULT_NAME}.
*/
String value() default DEFAULT_NAME;
/**
* Whether or not injection is required. Applicable only to methods and
* fields (not constructors or parameters).
*/
boolean required() default true;
}
关于Annotation的使用参见
http://www.cnblogs.com/peida/archive/2013/04/24/3036689.html
通过容器接口进行对象操作
1 getInstance
public class DefaultUnknownHandlerManager implements UnknownHandlerManager {
protected ArrayList<UnknownHandler> unknownHandlers;
private Configuration configuration;
private Container container;
@Inject
public void setConfiguration(Configuration configuration) {
this.configuration = configuration;
build();
}
@Inject
public void setContainer(Container container) {
this.container = container;
build();
}
/**
* Builds a list of UnknowHandlers in the order specified by the configured "unknown-handler-stack".
* If "unknown-handler-stack" was not configured, all UnknowHandlers will be returned, in no specific order
*/
protected void build() {
if (configuration != null && container != null) {
List<UnknownHandlerConfig> unkownHandlerStack = configuration.getUnknownHandlerStack();
unknownHandlers = new ArrayList<UnknownHandler>();
if (unkownHandlerStack != null && !unkownHandlerStack.isEmpty()) {
//get UnknownHandlers in the specified order
// 根据1定顺序获得UnknownHandler实例
for (UnknownHandlerConfig unknownHandlerConfig : unkownHandlerStack) {
//通过容器的getInstance方法取得容器内部脱光光的type为UnknownHandler.class
//name为传入的参数的UnknownHandler
UnknownHandler uh = container.getInstance(UnknownHandler.class, unknownHandlerConfig.getName());
unknownHandlers.add(uh);
}
} else {
//add all available UnknownHandlers
Set<String> unknowHandlerNames = container.getInstanceNames(UnknownHandler.class);
for (String unknowHandlerName : unknowHandlerNames) {
UnknownHandler uh = container.getInstance(UnknownHandler.class, unknowHandlerName);
unknownHandlers.add(uh);
}
}
}
}
.....
}
固然build方法的调用1定是在setContainer与setConfiguration调用以后。
2 inject
public class ActionSupport implements Action, Validateable, ValidationAware, TextProvider, LocaleProvider, Serializable {
private TextProvider getTextProvider() {
if (textProvider == null) {
TextProviderFactory tpf = new TextProviderFactory();
if (container != null) {
container.inject(tpf);
}
textProvider = tpf.createInstance(getClass(), this);
}
return textProvider;
}
@Inject
public void setContainer(Container container) {
this.container = container;
}
}
我们在下面可以看到container.inject(tpf)的参数,也就是TextProviderFactory里面有1个 @Inject标签。
固然getTextProvider之前得先调用setContainer。
public class TextProviderFactory {
private TextProvider textProvider;
@Inject
public void setTextProvider(TextProvider textProvider) {
this.textProvider = textProvider;
}
.....
}
感谢glt
参考资料
http://www.cnblogs.com/peida/archive/2013/04/24/3036689.html