`
kyo100900
  • 浏览: 639773 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

自己动手写一个Struts2

阅读更多

使用Struts2或webwork2有一段时间了,想把Struts2框架的思路简单的与大家分享一下,之前我是看过Struts2源代码的,所以本文算是它的一个功能非常有限的压缩版本。我也不打算重复发明轮子,只想让Struts2或Webwork2的新手更多的了解框架本身,而不仅仅是应用。废话少说,开始吧。

 

 

本文采用基本Xml来配置Action,如果有时间会继续写Annotation的实现。Xml文件结构与Struts2的配置文件struts.xml几乎一样,这样大家都比较熟悉,不过我简写了某些地方:

 

Struts.xml

<?xml version="1.0" encoding="UTF-8" ?>
<!-- 为简化框架,package的属性都没有实现;而且所有的元素都是简化版的 -->
<struts>
	<package>
		<action name="hello" method="hello"
			class="com.leo.action.HelloAction">
			<result name="success">/index.jsp</result>
		</action>
	</package>
</struts>

 

 

是不是很熟悉啊,不过为了简单我都给简化了,否则这是一个没有尽头的工作。核心部分仍然是Filter,Struts2所有工作都是通过一个Filter来完成的(struts1.*是通过一个Action实现的)。我们先来看代码:

 

public class StrutsFilter implements Filter {
	public void doFilter(ServletRequest req, ServletResponse res,
			FilterChain chain) throws IOException, ServletException {
		// TODO Auto-generated method stub
		HttpServletRequest request = (HttpServletRequest) req;
		HttpServletResponse response = (HttpServletResponse) res;
		ServletContext servletContext = filterConfig.getServletContext();
		// 解析Request的URL和传过来的参数
		String actionName = StringUtil.parseServletPath(request
				.getServletPath());

		// 如果后缀不为.action,那么直接放过,不进行拦截
		if (StringUtil.isEmpty(actionName)) {
			chain.doFilter(request, response);
		} else {
			// 解析得到ActionClass,里面包括Action的类全名,返回页面值,Action执行的方法
			ActionClass clas = this.getActionClass(actionName);
			// 得到页面的所有parameters参数(没考虑上传情况)
			Map<String, String[]> params = request.getParameterMap();
			// 为要调用的Action的set**方法设值,并返回要调用的Action对象本身
			setBeforeActionValue(clas, params);
			// 调用的Action执行方法,并返回值设置在request域中
			setResultValue(clas, request);
			// 返回相应的JSP页面
			servletContext.getRequestDispatcher(clas.getResult()).forward(
					request, response);
		}
	}
}

 

 没错一些因果都因doFilter方法而起。我的作法是:

 

  • 解析URL路径,从而得到相应action在struts.xml配置文件中所配置的方法。如果不为合法的action后缀,直接chain.doFilter(request, response)放行。
  • 将URL上的参数通过request.getParameterMap()取出,在调用action执行方法之前,将具有set**属性的字段赋值。这里利用了反射。
  • 继续利用反射,执行Action的方法。结束前将具有get**属性的字段存于request域中,供页面使用。大家记得在每次执行Action的方法时都会返回一个String字符串,比如SUCCESS,INPUT,ERROR等,我们可以将这个值与struts.xml的<result name="success" ....>/index.jsp</result> 结点进行匹配,取出返回页面/index.jsp
  • 一切准备就绪后,调用servletContext.getRequestDispatcher().forward()方法到相应的页面上去。

这就是大概的流程。因为是一个入门的框架所以很不完善,拦截器,result type的各种类型都没有去实现,因为我压根没想过要重复发明轮子。好,我们开始一步一步的看。

 

 

 

我们先来看这一句:ActionClass clas = this.getActionClass(actionName);
其中ActionClass的结构如下:

 

public class ActionClass {
	/**
	 * 类名
	 */
	private String className;
	/**
	 * 要调用的方法名
	 */
	private String method;
	/**
	 * 返回结果页面
	 */
	private String result;
	/**
	 * 临时存储Action下的所有result结点
	 */
	private List<Element> elements = new ArrayList<Element>();

	/**
	 * 要调用的Action本身
	 */
	private Object action;

	//省略所有的set,get方法
} 

 

ActionClass主要是用来存放解析struts.xml文件一些有用的值,以及反射时所调用的Action本身对象,其实就是一个简单的JavaBean,存储临信息。

 

 

getActionClass(actionName)方法就是将URL上的actionName取出与struts.xml中的<action>结点匹配,可以得到Action的类全名,Action所调用的具体哪个方法名,Action的所有result结点(因为方法还没有执行,还不知道是具体哪一个result结点,所以先存起来,后面来解析)分别存在ActionClass对象中相应的属性中去。具体的解析代码我就不打出来了,否则文章太长。

 

 

再来看看setBeforeActionValue(clas, params)这一句。其实就是将得到的ActionClass对象与提交的参数全部传进去,给Action的那些set属性赋值:

 

/**
	 * 调用Action,并执行Action的无参方法
	 * 
	 * @param actionClass
	 * @param request.getParameterMap()
	 * @return
	 */
	public Object setActionValues(ActionClass actionClass,
			Map<String, String[]> params) {
		try {
			// 得到Action的Class,并根据无参构造函数生成一个Action对象
			Class clas = Class.forName(actionClass.getClassName());
			Object obj = clas.newInstance();

			if (params != null && params.size() > 0) {
				Iterator<String> it = params.keySet().iterator();
				while (it.hasNext()) {
					String key = it.next();
					String[] value = params.get(key);
					String upperFirstLetter = key.substring(0, 1).toUpperCase();
					// 获得和属性对应的setXXX()方法的名字
					String setMethodName = "set" + upperFirstLetter
							+ key.substring(1);
					Method method = null;
					// 看看该页面提交的参数名中,是否在Action有set方法
					try {
						method = clas.getMethod(setMethodName,
								new Class[] { String.class });
					} catch (NoSuchMethodException e) {
						System.out.println("警告 " + actionClass.getClassName()
								+ "." + setMethodName + "("
								+ String.class.getName() + ") 不存在");
					}
					if (method != null) {
						// 如果有set方法,就调用set方法,进行赋值操作
						String result = StringUtil.StringArrayToString(value);
						method.invoke(obj, new String[] { result });
					}

				}

			}
			return obj;
		......
	}

 

 

这样就顺利的将页面的值赋给了Action的相应属性,接下来就是Action调用工作了。通过setActionValues方法,我们已经可以得到Action对象本身了,可以存在ActionClass对象clas中去,我们直接调用setResultValue(clas, request)在Action执行后,同时也把有get方法的属性一并存于request域中:

 

 

/**
	 * 调用Action,并执行Action的无参方法
	 * 
	 * @param actionClass
	 * @param obj
	 *            要处理的对象
	 * @return
	 */
	public Object invokeAction(ActionClass actionClass) {
		try {
			Object obj = actionClass.getAction();
			Class clas = obj.getClass();
			Method method = clas.getMethod(actionClass.getMethod(), null);
			String result = (String) method.invoke(obj, null);
			this.setInvokeResult(result, actionClass);
			actionClass.setAction(obj);
			return obj;
                                                ......
		}
	}

 

很简单的代码——调用Action那个无参执行方法,得到返回String类型的返回结果,然后我们可以再次利用ActionClass将最终的返回结果也给解析出来,看this.setInvokeResult(result, actionClass)方法

 

 

/**
	 * 匹配<result name="success">/index.jsp</result> Xml中的result
	 * name属性,如果匹配成功,设置返回结果"/index.jsp"
	 * 
	 * @param result
	 * @param actionClass
	 */
	private void setInvokeResult(String result, ActionClass actionClass) {
		List<Element> elements = actionClass.getElements();
		for (Element elem : elements) {
			Attribute name = XmlUtil.getAttributeByName(elem, "name");
			if (StringUtil.equals(result, name.getText())) {
				actionClass.setResult(elem.getText());
				return;
			}

		}
		throw new RuntimeException("请确定在xml配置文件中是否有名叫 [" + result
				+ "] 的返回类型结点 ");
	}

 

 

 

一切大功告成,将刚刚得到的返回结果用servletContext.getRequestDispatcher(clas.getResult()).forward(
request, response)转发出去,编码部分完毕。

 

 

别忘记了在web.xml中配置这个Filter:

 

 

<filter>
	<filter-name>struts</filter-name>
	<filter-class>com.framework.core.StrutsFilter</filter-class>
</filter>
<filter-mapping>
	<filter-name>struts</filter-name>
	<url-pattern>/*</url-pattern>
</filter-mapping>

 

 

 

 

最后写个测试Action吧,就按照本文最开始的那个struts.xml配置编写HelloAction.java

 

 

public class HelloAction {

	private String message;

	public String hello() {
		message = "superleo " + this.message;
		return "success";
	}

	public String getMessage() {
		return message;
	}

	public void setMessage(String message) {
		this.message = message;
	}

}

 

 

 

 

 

 

如果下载了源代码,可分别输入下列几个链接看看测试效果:(test文件夹下是单元测试)

  • framework.rar (413.6 KB)
  • 描述: 本文的源代码。
  • 下载次数: 1155
62
13
分享到:
评论
10 楼 zmxt2008 2008-08-15  
写的很好呀,高手 学习中。
9 楼 kaka99 2008-08-15  
很好的东东,很容易看懂!
8 楼 kevindurant 2008-08-15  
哦..学习学习..
  俺还是菜鸟..不知道楼上各位说的:"重复发明轮子"啥意思??用这个框架是重复发明轮子吗???
 
    望得指点!!
7 楼 zhenjia 2008-08-14  
呵呵 这很有意思 我用WEBWORK2年了,虽然说它很不错,但是由于公司项目遇到的种种原因,我也自己重复发明轮子,和你一样自己重新写了一个。脱离SPRING IOC的支持,用到了guice1.0.虽然struts2.0也有个guice plugin但是那东西是在第一次调用action时候加载的。在用guice的过程中有很多很多的不方便。所以也把这给替换了,没有了xml配置
基本形式如下
@GSAction(name = "cpny", namespace = "/hr")
public class SysCompanyAction extends ActionSupport<SysCompany> {

@Inject
private SysCompanyService sysCompanyService;

@ReqGet
private Long id;



@ReqGet
@ModelDriver
@ReqSet
private SysCompany sysCompany;

@PageFlow(result = { @Result(name = "success", path = "/view/sys/company.jsp", type = Dispatcher.Forward) })
public String execute() throws Exception {
return "success";
}
}
给您点建议
我不知道您现在在Action里的setter方法 是不是支持任何object
我看你demo是String message 如果换成User user能赋值吗
user.id=1&user.name=zhenjia这样形式的传参
还有User 有个Department对像 也要user.department.id等等
这里建议采用ognl来赋值
目前我自己完成的这小玩意用Annotation来做的
帖下部分Annotation
/**
User:zhenjia(zhenjiaWang@gmail.com)
Date:2008-8-5
Time:下午04:42:02
<p>
该注释 运用在FIELD之上
FILED如果引用@ReqGet则会自动进行Inject
Inject值会根据当前FILED TYPECLASS进行自动转换 转换失败值为NULL
该注释带有一个参数(value) default "";
不显示声明value,value值为FILED显示声明name,根据驼峰规则首字母小写
在Inject时根据注释value 作为key从HttpServletRequest获取参数

@ ReqGet String empId;等同于request.getParameter("empId");
@ ReqGet("empId") String id;等同于request.getParameter("empId");
</p>
**/
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface ReqGet {
String value() default "";
}



/**
User:zhenjia(zhenjiaWang@gmail.com)
Date:2008-8-7
Time:下午01:55:33
<p>
该注释 运用在FIELD之上
FILED如果引用@ReqSet则会自动进行request.setAttribute
该注释带有一个参数(value) default "";
不显示声明value,value值为FILED显示声明name,根据驼峰规则首字母小写
在setAttribute时根据注释value 作为key将FILED添加到当前requestScope

@ ReqSet String empId;等同于request.setAttribute("empId",empId);
@ ReqSet("empId") String id;等同于request.setAttribute("empId",id);
</p>
**/
@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface ReqSet {
String value() default "";
}

/**
User:zhenjia(zhenjiaWang@gmail.com)
Date:2008-8-1
Time:下午04:03:02
<p>
该注释 运用在FIELD之上 与@ReqSet配合使用
可将参数自动封装为JavaBean 如参数里没有任何一个属性匹配该对象则返回null
如:
@ ReqSet @ ModelDriver Employee employee
request参数如果包含
employee.id,employee.empName,employee.dept.id,employee.dept.deptName等
会自动将4个参数自动封装在employee对像里 自动构造关联对象
带参数如:
@ ReqSet @ ModelDriver("emp") Employee employee
request参数如果包含
emp.id,emp.empName,emp.dept.id,emp.dept.deptName等
会自动将4个参数自动封装在employee对像里 自动构造关联对象
</p>
**/
@Target({ElementType.FIELD,ElementType.PARAMETER})
@Retention(RetentionPolicy.RUNTIME)
public @interface ModelDriver {
String value() default "";
}

其实我不太喜欢@ReqSet 但是太忙也太懒了,我比较喜欢WEBWORK的OgnlValueStack 但是自己实现以后需要自己写标签,用jstl体现不出它的好来。以后等有机会就把这点遗憾给补上
另WEBWORK的基石Interceptor我也实现了 当然是用guice来做的。
这应该是你手写STRUTS2.0的第一步。以后肯定会越来越完善的,我一直在用WEBWORK2 不知道STRUTS2.0于WEBWORK2相比有没有更好的东西。如果有请告诉我。
6 楼 phz50 2008-08-14  
   
5 楼 melin 2008-08-14  
给set方法设置值。我估计是用了ONGL、猜的,没有看过代码
4 楼 silen85 2008-08-14  
思路好清晰哦,可以理解大概流程,要我写就有点困难了! 
3 楼 silen85 2008-08-14  
思路好清晰哦,可以理解大概流程,要我写就有点困难了! 
2 楼 troyconder 2008-08-14  
很有见地 说明你已经理解了Struts2 其实框架就是基础代码+模式的集合
1 楼 clz1314521 2008-08-14  
不错!不错

相关推荐

    自己动手写一个Struts2(升级版)

    这篇“自己动手写一个Struts2(升级版)”的博客文章可能是作者对Struts2框架进行的深入解析,旨在帮助读者理解其内部工作原理,通过实践来提升编程技巧。 首先,我们来看一下Struts2的核心概念: 1. **Action类**:...

    自己动手写STRUTS 源代码

    动手实践Struts源代码的编写,不仅可以提升对Struts框架的理解,还能加深对MVC模式的认识,对于Java Web开发者来说,这是一个非常有价值的学习经历。通过实际操作,你将能够更好地应对项目开发中的挑战,提高问题...

    自己动手写一个struts1

    自己动手写Struts1 要实现一个简易的Struts1框架,你需要完成以下步骤: 1. **创建ActionServlet**:实现Servlet接口,处理HTTP请求。这里需要解析请求参数,根据ActionMapping找到对应的Action执行。 2. **编写...

    自己动手写Struts.pdf

    ### 自己动手写Struts:构建基于MVC的Web开发框架 #### 第一篇:Web框架入门 ##### 第1章 运筹帷幄:Web框架的核心思想 - **MVC模式** - **MVC模式概述**:MVC(Model-View-Controller)是一种常用的设计模式,...

    ISO文件《自己动手写Struts随书光盘》

    ISO文件《自己动手写Struts随书光盘》,和大家分享~

    自己动手写Struts:构建基于MVC的Web开发框架

    《自己动手写Struts》一书通过具体的案例,教授读者如何从零开始构建一个基于MVC的Web开发框架。这一过程不仅能加深对Struts框架的理解,还能帮助开发者更好地掌握MVC设计模式的核心思想。 #### 5. 构建框架的关键...

    struts2 demo project

    这个"struts2 demo project"是为了帮助开发者理解和实践Struts2框架而创建的一个示例项目,旨在提供一个基础的、可运行的实例,让大家能够深入探索Struts2的核心特性和功能。 在Struts2框架中,以下是一些关键知识...

    Struts2示例demo

    Struts2是一个强大的Java web应用程序框架,用于构建和部署可维护、高性能的Web应用程序。它在Struts1的基础上进行了很多改进,提供了更优秀的MVC(Model-View-Controller)架构支持,增强了动作处理、异常处理、...

    struts2框架的使用

    Struts2是一个强大的Java web开发框架,它基于MVC(Model-View-Controller)设计模式,为构建可维护、可扩展...记住,实践是检验理论的最好方式,尝试自己动手创建一个简单的Struts2项目,将会加深对这些知识点的理解。

    一个超级简单的struts2结构

    Struts2是一个强大的Java web应用程序框架,用于构建和部署可维护、高性能的Web应用程序。它在Web开发领域中被广泛使用,特别是在企业级应用中,因为它提供了MVC(Model-View-Controller)设计模式的实现,简化了...

    Struts2小demo

    Struts2是一个强大的Java web应用程序框架,用于构建和部署可维护、高性能的Web应用程序。它在原有的Struts1基础上进行了很多改进,引入了更多现代化的开发理念和技术,如依赖注入(DI)、面向切面编程(AOP)以及...

    struts2教学用ppt

    Struts2是一个基于MVC(Model-View-Controller)设计模式的Java web应用程序框架,它在Java社区中广泛使用,特别是在开发企业级应用时。这个"struts2教学用ppt"很可能是为了帮助学习者理解并掌握Struts2的核心概念、...

    struts2初学者第一个实例

    对于初学者来说,理解并动手实践Struts2的第一个实例是非常重要的,这将帮助他们快速掌握该框架的基本用法。下面,我们将详细讲解如何创建并运行一个简单的Struts2应用实例。 首先,你需要确保已经安装了Java SDK和...

    详细的Struts2教程

    Struts2是一个基于MVC(Model-View-Controller)设计模式的开源Java Web框架,它在Struts1的基础上进行了很多改进和增强,旨在简化企业级Web应用的开发。本教程将深入探讨Struts2的核心概念、特性以及如何使用它来...

    struts2的学生管理系统

    Struts2是一个强大的Java web应用程序框架,用于构建和部署可维护、高性能的企业级应用程序。它在MVC(Model-View-Controller)设计模式的基础上提供了一种结构化的方式来组织和管理应用代码,使得开发者能够更专注...

    struts2官方自带的四个例子

    Struts2是一个强大的Java web应用程序框架,它基于MVC(Model-View-Controller)设计模式,为开发者提供了构建可维护性、可扩展性良好的应用程序的工具。本篇将详细解析Struts2官方自带的四个例子,帮助你深入理解其...

Global site tag (gtag.js) - Google Analytics