Struts2中action接收参数的三种方法及ModelDriven跟Preparable接口结合JAVA反射机制的灵活用法

Struts2中action接收参数的三种方法及ModelDriven和Preparable接口结合JAVA反射机制的灵活用法
Struts2中Action接收参数的方法主要有以下三种:

1.使用Action的属性接收参数(最原始的方式):
    a.定义:在Action类中定义属性,创建get和set方法;
    b.接收:通过属性接收参数,如:userName;
    c.发送:使用属性名传递参数,如:user1!add?userName=jim;
2.使用DomainModel接收参数:
    a.定义:定义Model类,在Action中定义Model类的对象(不需要new),创建该对象的get和set方法;
    b.接收:通过对象的属性接收参数,如:user.getUserName();
    c.发送:使用对象的属性传递参数,如:user2!add?user.userName=mike;
3.使用ModelDriven接收参数(现在用的比较多的方式):
    a.定义:Action实现ModelDriven泛型接口,定义Model类的对象(必须new),通过getModel方法返回该对象;
    b.接收:通过对象的属性接收参数,如:user.getUserName();
    c.发送:直接使用属性名传递参数,如:user2!add?userName=tom

在Struts2.3.4的文档里面有这样说明:
To use ModelDriven actions, make sure that the Model Driven Interceptor is applied to your action. This interceptor is part of the default interceptor stack defaultStack so it is applied to all actions by default.

Action class:
public class ModelDrivenAction implements ModelDriven { 
    public String execute() throws Exception {
        return SUCCESS;
    }

    public Object getModel() {
        return new Gangster();
    }
}

JSP for creating a Gangster:
<s:form action="modelDrivenResult" method="POST" namespace="/modelDriven">   
    <s:textfield label="Gangster Name" name="name" />
    <s:textfield label="Gangster Age"  name="age" />
    <s:checkbox  label="Gangster Busted Before" name="bustedBefore" />
    <s:textarea  cols="30" rows="5" label="Gangster Description" name="description" />           
    <s:submit />
</s:form>


在Model Driven Interceptor里面这样说道:
To create a Model Driven action, implement the ModelDriven interface by adding a model property, or at least the accessor.

public Object getModel() ...

In the implementation of getModel, acquire an instance of a business object and return it.

On the page, you can address any JavaBean properties on the business object as if they were coded directly on the Action class. (The framework pushes the Model object onto the ValueStack.)

Many developers use Spring to acquire the business object. With the addition of a setModel method, the business logic can be injected automatically.

所以如果实现 ModelDriven 接口,那么必须至少构造一个getModel方法,并return一个实体对象。而且在struts.xml文件中需要配置名为modelDriven的拦截器Interceptor,如果没有指定拦截器栈,那么使用默认的defaultStack,这个拦截器栈里面已经引用了modelDriven的拦截器,所以默认下你的package包extends了struts-default那么就不用配置。
struts.xml:
<action name="someAction" class="com.examples.SomeAction">
    <interceptor-ref name="modelDriven"/>
    <interceptor-ref name="basicStack"/>
    <result name="success">good_result.ftl</result>
</action>


Struts2默认的拦截器栈:
            <!-- A complete stack with all the common interceptors in place.
                 Generally, this stack should be the one you use, though it
                 may do more than you need. Also, the ordering can be
                 switched around (ex: if you wish to have your servlet-related
                 objects applied before prepare() is called, you'd need to move
                 servletConfig interceptor up.

                 This stack also excludes from the normal validation and workflow
                 the method names input, back, and cancel. These typically are
                 associated with requests that should not be validated.
            -->
            <interceptor-stack name="defaultStack">
                <interceptor-ref name="exception"/>
                <interceptor-ref name="alias"/>
                <interceptor-ref name="servletConfig"/>
                <interceptor-ref name="i18n"/>
                <interceptor-ref name="prepare"/>
                <interceptor-ref name="chain"/>
                <interceptor-ref name="scopedModelDriven"/>
                <interceptor-ref name="modelDriven"/>
                <interceptor-ref name="fileUpload"/>
                <interceptor-ref name="checkbox"/>
                <interceptor-ref name="multiselect"/>
                <interceptor-ref name="staticParams"/>
                <interceptor-ref name="actionMappingParams"/>
                <interceptor-ref name="params">
                         <param name="excludeParams">dojo\..*,
                         ^struts \..*,^session\..*,^request\..*,^application\..*,
                         ^servlet(Request|Response)\..*,parameters\...*
                         </param>
                </interceptor-ref>
                <interceptor-ref name="conversionError"/>
                <interceptor-ref name="validation">
                    <param name="excludeMethods">input,back,cancel,browse</param>
                </interceptor-ref>
                <interceptor-ref name="workflow">
                    <param name="excludeMethods">input,back,cancel,browse</param>
                </interceptor-ref>
                <interceptor-ref name="debugging"/>
            </interceptor-stack>
......
        <default-interceptor-ref name="defaultStack"/>

        <default-class-ref class="com.opensymphony.xwork2.ActionSupport" />



下面这段代码可以充分体现ModelDriven和Preparable接口结合JAVA反射机制的灵活用法,大大方便节省了开发code时间,注意这里用的是paramsPrepareParamsStack拦截器栈,所以params拦截器会在Preparable接口的方法之前执行。Struts2中action接收参数的三种方法及ModelDriven跟Preparable接口结合JAVA反射机制的灵活用法
public class PlayerAction extends AbstractBaseAction<Player> {

	private static final long serialVersionUID = -3068068486865209475L;
	
	@Override
	public String execute() throws Exception {
		
		return super.list();
	}

}


public abstract class AbstractBaseAction<T> extends ActionSupport implements ModelDriven<T>,Preparable{

	private static final long serialVersionUID = -1487318639557604204L;
	
	private T entity;
	
	private Class<T> entityClass;
	
	private Long id;
	
	public T getModel() {
        return entity;
    }
	
	@SuppressWarnings("unchecked")
	public AbstractBaseAction() {
		try {
            entityClass =(Class<T>)((ParameterizedType)getClass().getGenericSuperclass()).getActualTypeArguments()[0];
        } catch (Exception e) {
            ExceptionHandler.logError(e, AbstractBaseAction.class,"无法获取entityClass ");
        }
	}
	
	/**
	 * 这里相当于new了一个entity对象 ,getModel()才能返回这个对象,
     * 因为默认的interceptor-stack是defaultStack中prepare在modelDriven拦截器之前执行
	 */
	@Override
	public void prepare() throws Exception {
		if (entity == null) {
	        try {
	            entity = entityClass.newInstance();
	        } catch (Exception e) {
	            ExceptionHandler.logError(e, AbstractBaseAction.class,"创建实例失败,class=" + entityClass.getName());
	        }
	    }
	}
	
	public void prepareSave() throws Exception {
		if (getId() != null) {
			entity = getEntityById(getId());
		}
	}
	
	public String save() throws Exception{
		return SUCCESS;
	}
	
	public void prepareDelete() throws Exception {
		if (getId() != null) {
			entity = getEntityById(getId());
		}
	}
	
	public String delete() throws Exception{
		return SUCCESS;
	}
	
	public void prepareLoad() throws Exception {
		if (getId() != null) {
			entity = getEntityById(getId());
		}
	}
	
	public String load() throws Exception{
		return SUCCESS;
	}
	
	public String list() throws Exception{
		return SUCCESS;
	}
	
    protected T getEntityById(Long id) {
    	if (id!=null) {
    		try {
                return (T) HibernateUtil.getObjectById(entityClass, id);
            } catch (Exception e) {
            	Struts2Utils.getRequest().setAttribute("error", "数据查询出错");
            	addActionMessage("数据查询出错");
                ExceptionHandler.logError(e, AbstractBaseAction.class,"获取实例时出现异常,class=" + entityClass.getName() + ", id=" + getId());
            }
		}
        return null;
    }

	public Long getId() {
		return id;
	}

	public void setId(Long id) {
		this.id = id;
	}

}