Struts2模型驱动核心机制详解

Struts2采用模型驱动来实现对于表单的多个属性进行封装并通过调用action中的set方法来注入到action中对应的POJO.

首先,Struts2中包括属性注入和域模型注入,属性注入正如我们所熟悉的servlet中对于每个表单提交过来的参数进行request的getParameter方法,

获取每个属性值来注入类中。域模型注入则是上面我们所说的属性封装,下面我们看个例子:

首先定义jsp页面中的表单:

 <form action="hello.action" method="post">
        用户名:<input type="text" name="user.userName"/><br/><br/>
        密码:<input type="text" name="user.password"/><br/><br/>
        <input type="submit" value="提交"/>
 </form>    


再来看下我们定义的action类:

import com.opensymphony.xwork2.ModelDriven;
import entity.User;
public class HelloAction {
    //定义实体对象属性,接收表单参数:用户名、密码
    private User user;
    public void setUser(User user) {
        System.out.println("注入对象user...");
        this.user = user;
    }
    
    public User getUser() {
        return this.user;
    }
    /**
     * 在业务方法中输出“Hello,Action.”
     */
    public String execute() {
        System.out.println("Hello,Action.");
        // 输出基本类型数据
        
        // 输出域模型方式注入的参数
        System.out.println("用户名:" + user.getUserName());
        System.out.println("密码:" + user.getPassword());
        return "success";
    }
}
我们可以看到,struts2通过ModelDriven类自动将user.userName和user.passWord封装进action中的user对象,然后struts2调用默认的execute方法来执行之后的操作,那么模型驱动是如何完成这一封装机制的呢,我们就通过下面这个例子讲解一下:

我么使用servlet来模拟这一封装操作,这里用到了java的反射机制:

LoginServlet类:

import java.io.IOException;
import java.io.PrintWriter;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import com.pojo.Users;

public class LoginServlet extends BaseServlet {
	//http://www.jikexueyuan.com/course/530.html
	
	private Users us = new Users();
	
	
	public String execute() throws Exception {
		System.out.println("自动封装数据后的结果:");
		System.out.println(us.getUsername());
		System.out.println(us.getPassword());
		return "/index.jsp";
	}

	public Users getModel() {
		// TODO Auto-generated method stub
		return us;
	}
	

}
这里的LoginServlet继承了BaseServlet,并且定义并实例化了pojo:Users,Users包含两个属性:username和password,并且通过getModel方法我们可以返回实例化的属性,我们再来看看BaseServlet

BaseServlet:

import java.io.IOException;
import java.io.PrintWriter;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.util.Enumeration;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServlet;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

public class BaseServlet extends HttpServlet {

	public void doGet(HttpServletRequest request, HttpServletResponse response)
			throws ServletException, IOException {
		doPost(request, response);
		
	}
	public void doPost(HttpServletRequest request, HttpServletResponse response)
			throws ServletException, IOException {
		try {
			//在程序运行时 获得当前类的getModel方法对象
			Method method = this.getClass().getDeclaredMethod("getModel", null);
			//通过invoke方法 调用该方法 获得实体类对象
			Object ob = method.invoke(this, null);
			//通过实体类对象获得  类类型
			Class cl = ob.getClass();
			//通过类类型获得 类中的属性对象数组
			Field[] fi = cl.getDeclaredFields();
			//通过request获得所有的 表单中提交的name值 
			Enumeration em = request.getParameterNames();
			//循环枚举中的值
			while(em.hasMoreElements()){
				//获得枚举中的值  就是表单提交的name值
				String fieldName = em.nextElement().toString();
				System.out.println("从表单中获得的 name值;"+fieldName);
				//循环类中所有的属性对象
				for(int i =0;i<fi.length;i++){
					System.out.println("从类中获得的属性:"+fi[i].getName());
					//判断 如果类中属性的名字 和 表单中提交的名字一致
					if(fieldName.equals(fi[i].getName())){
						fi[i].setAccessible(true);//原本对象属性为private,可以被外部赋值
						//获取request.getParameter(fieldName) 封装到对象中
						fi[i].set(ob, request.getParameter(fieldName));
						
					}
				}
			}
			//调用默认的方法 执行处理
			Method me1 = this.getClass().getDeclaredMethod("execute", null);
			//获得返回的url字符串
			Object url = me1.invoke(this, null);
			//根据返回的字符串 跳转页面
			request.getRequestDispatcher(String.valueOf(url)).forward(request, response);
			
		} catch (Exception e) {
			e.printStackTrace();
		}
			
		
	
	}

}
这里我们模拟了这一数据封装操作,即

1)当请求发送到action之前,调用loginServlet类中getModel() 获取要将表单数据封装到哪个实例化的对象中
2)获得到该对象之后,我们可以获得 类类型
3)获得类 类型之后。获得类中的属性request.getParameters获得 表单提交的所有数据名从而获取值
4)如果表单提交的 name值 与 实体类中属性名一致那么我们将获得表单中的数据 封装到us对象当中去
5)最后调用servlet类中的execute执行具体action逻辑

总结:

Struts2中的模型驱动是通过反射机制来实现POJO属性的自动封装和注入的。


  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值