XWork中的容器1

本文是<<struts2 技术内幕>>的学习笔记

在进行面向对象编程的时候,我们不可避免地要使用继承实现等等java提供的语法支持。但是复杂的对象关系也为对象生命周期的管理带来了至少以下两个问题。

1 程序运行时,应如何双肩我们所需要的对象。

2 当创建一个对象后,如何保证与其相关联的依赖关系也正确的被创建处理。

好在先辈们已经给我们想好了出路------在程序中引入一个额外的编程元素:容器(Container)

对象的生命管理周期

首先我们得引入一个概念-----控制反转(Inverse of Control)

什么是控制反转?从字面上来说大概就是

本来应该由我控制的事情,不再需要我来控制了。至于交给谁来控制,我不需要知道。

举个例子。

public class Person{
    private Car car;

    public Person(Car c){
    this.car=c;
    }
    public void drive(){
    car.drive();
    }
}

上面的例子很简单吧,一个人能开车。对于车的产生,需要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容器的定义

刚才一直再说容器,现在我们就说说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();
}

接口中的方法可以分为三类

获取对象实例 getInstance,getInstacneNames

处理对象依赖关系 inject

处理对象的作用范围策略 setScopeStrategy,removeScopeStrategy

并且,容器是一个辅助的编程元素,它在系统中被设计为一个全局的单例的对象。

XWork容器的管辖范围

刚才我们已经初步认识了XWork容器,现在我们看看:XWork到底能管理哪些东西。

1 获取对象实例

getInstance可以获得被容器托管的对象。那么到底哪些对象是容器所托管的呢?

看struts-default.xml里面有三类元素,是容器所托管的

1在bean节点中声明的框架内部的对象

2在bean节点中声明的自定义对象

3在constant节点中声明的系统运行参数

另外还有

在Properties文件里什么歌的系统运行参数

再换句话说,想把自定义的对象纳入容器的管理范围,只需在Sturs/XWork的配置文件里声明即可。

2 对象的依赖注入

就像我们上面说的people与car的关系,people里面需要一个car,而car是通过容器所管理的,那么对people的注入就是container.inject(people)。

inject方法的参数都可以使什么呢?

调用XWork容器的inject方法,能够帮助我们将容器所管理的对象(包括框架的内部元素及系统运行参数)注入到任意的对象实例中。从而建立起任意对象与框架元素沟通的桥梁。

那么系统怎么知道people这个对象里面就需要一个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
        // 根据一定顺序获取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方法的调用一定是在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里面有一个 @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

时间: 2024-08-10 23:28:47

XWork中的容器1的相关文章

【转】C++中map容器的说明和使用技巧

C++中map容器提供一个键值对容器,map与multimap差别仅仅在于multiple允许一个键对应多个值. 一.map的说明    1   头文件   #include   <map>     2   定义   map<string,   int>   my_Map;   或者是typedef     map<string,   int>   MY_MAP;   MY_MAP   my_Map;     3   插入数据   (1)   my_Map["

初识Java中的容器

记得第一次听到java中的容器是一个师哥说的,当时听着十分神秘.那么今天就来揭开那层神秘的面纱. 首先什么是容器呢? 在书写程序的时候,我们常常需要对大量的对象引用进行管理.为了实现有效的归类管理,我们常常将同类的引用放置在同一数据容器中.由于数据容器中存放了我们随时可能需要使用到的对象引用,所以一般的数据容器要都要能能提供方便的查询.遍历.修改等基本接口功能. 早期的OOP语言都通过数组的方式来实现对引用集的集中管理和维护. 但是数组方式下,数组大小需要提前被确定,并不允许修改大小,导致其作为

理解Docker(6):若干企业生产环境中的容器网络方案

本系列文章将介绍 Docker的相关知识: (1)Docker 安装及基本用法 (2)Docker 镜像 (3)Docker 容器的隔离性 - 使用 Linux namespace 隔离容器的运行环境 (4)Docker 容器的隔离性 - 使用 cgroups 限制容器使用的资源 (5)Docker 网络 (6)若干企业生产环境中的容器网络方案 Docker 在早期只有单机上的网络解决方案,在 1.19 版本引入了原生的 overlay 网络解决方案,但是它的性能损耗较大,可能无法适应一些生产环

在libvirt 中体验容器

libvirt是一个通用的虚拟化框架,支持xen,kvm,lxc多种虚拟化技术,本文作为一个笔记. 设置LXC为默认(默认为qemu) export LIBVIRT_DEFAULT_URI=lxc:/// virsh -c lxc:/// 可省略 网络设置 NAT NAT是libvirt默认自带,叫做default,通过下面的 命令查看 virsh net-list 假设宿主机有两个网卡,eth0为内网,eth1为外网,我们可以为容器桥接出两个接口: #cat lan.xml <network>

STL中的容器

STL中的容器 一. 种类: 标准STL序列容器:vector.string.deque和list. 标准STL关联容器:set.multiset.map和multimap. 非标准序列容器slist和rope.slist是一个单向链表,rope本质上是一个重型字符串 非标准关联容器hash_set.hash_multiset.hash_map和hash_multimap. 几种标准非STL容器,包括数组.bitset.valarray.stack.queue和priority_queue 值得

java中的容器解释

解释一:容器(Container)Spring 提供容器功能,容器可以管理对象的生命周期.对象与对象之间的依赖关系,您可以使用一个配置文件(通常是XML),在上面定义好对象的名称.如何产生(Prototype 方式或Singleton 方式).哪个对象产生之后必须设定成为某个对象的属性等,在启动容器之后,所有的对象都可以直接取用,不用编写任何一行程序代码来产生对象,或是建立对象与对象之间的依赖关系.换个更直白点的说明方式:容器是一个Java 所编写的程序,原先必须自行编写程序以管理对象关系,现在

C++中的容器

C++中的容器 容器与容器适配器 容器包括vector, deque, list, map, multimap, set, multiset.容器适配器包括基于deque的stack和queue,基于vector的priority_queue.string也实现了stl的接口. 因为编写C++程序时经常需要查找容器的函数接口,故作此总结.C++新引入的容器与函数未引入.主要参考自:STL Containers and Container Adaptors 序列容器 包括vector,deque,

如何让不同主机中docker容器可以相互访问

我们都知道同一台主机中的docker容器可以相互访问,那么不同主机中的docker容器间是否可以相互访问呢?答案是可以的,解决思路是在主机中把另外主机上docker容器的网段加入到路由表中. 现有主机A 192.168.124.51,主机B 192.168.124.55,具体步骤如下: 修改一台主机docker容器使用的网段(默认是172.17.x.x),如何修改请看上一篇. 假设已经修改好docker网段,例如主机A中docker容器使用10.1.x.x网段,主机B中docker使用172.1

docker中的容器互联-linking系统

docker中的容器互联-linking系统docker有一个linking 系统可以连接多个容器.它会创建一对父子关系,父容器可以看到所选择的子容器的信息.1)容器的命名系统linking系统依据容器的名称来执行.当我们创建容器的时候,系统会随机分配一个名字.当然我们也可以自己来命名容器,这样做有2个好处:• 当我们自己指定名称的时候,比较好记,比如一个web应用我们可以给它起名叫web• 当我们要连接其他容器时候,可以作为一个有用的参考点,比如连接web容器到db容器使用--name标记可以