早在struts2.0.*的时候,struts2的必备jar包需要如下几个:
commons-logging-*.jar Apache旗下commons项目的log日志包
freemarker-*.jar 一种前台页面模板,应用比较广泛
ognl-*.jar 动态图导航语言,struts2处理前台页面的核心语言,相当实用
struts2-core-*.jar struts2的核心包
xwork-core-*.jar webwork的核心包,因为struts2的前身是webwork,所以这个是必须的
只要满足这5个基本jar包就可以运行一个struts2的hello world
(注:*表示版本号,struts2不同版本,使用的jar包版本也可能不一样。)
但是后面版本,比如struts-2.1.8.1,必须再添加一个新jar包才能运行,不然启动就会报错:
commons-fileupload-1.2.1.jar 支持文件上传的jar包
<!-- struts2 滤镜配置 --> <filter> <filter-name>struts2</filter-name> <filter-class>org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter</filter-class> <init-param> <param-name>config</param-name> <param-value> struts-default.xml, struts-plugin.xml, <!-- 核心struts.xml文件 --> ../conf/common/struts2/struts.xml, </param-value> </init-param> </filter> <filter-mapping> <filter-name>struts2</filter-name> <url-pattern>/*</url-pattern> </filter-mapping>
<?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE struts PUBLIC "-//Apache Software Foundation//DTD Struts Configuration 2.0//EN" "http://struts.apache.org/dtds/struts-2.0.dtd"> <struts> <!--载入默认的struts配置--> <include file="struts-default.xml" /> <!--指定web应用的默认编码集,相当于调用HttpServletRequest的setCharacterEncoding方法--> <constant name="struts.i18n.encoding" value="UTF-8"></constant> <!--该属性指定需要Struts 2处理的请求后缀,该属性的默认值是action,即所有匹配*.action的请求都由Struts 2处理 如果用户需要制定多个请求后缀,则多个后缀之间以英文逗号隔开--> <constant name="struts.action.extension" value="action,do"></constant> <!--设置浏览器是否缓存静态内容,默认值为true,生产环境下使用,开发阶段最好关闭 --> <constant name="struts.serve.static.browserCache" value="false"></constant> <!--当struts的配置文件修改后,系统是否自动重新加载该文件,默认值为false,生产环境下使用,开发阶段最好打开 --> <constant name="struts.configuration.xml.reload" value="true"></constant> <!--开发模式下使用,可以打印出更详细的错误信息 --> <constant name="struts.devMode" value="true" /> <!-- 动态方法调用 false为不允许 --> <constant name="struts.enable.DynamicMethodInvocation" value="true" /> <!-- 默认的视图主题,标签不支持label ; theme属性包括xhtml,html,simple,ajax ,默认是xhtml--> <constant name="struts.ui.theme" value="simple"></constant> <!--Struts2集成Spring:所有action对象由Spring来负责创建--> <constant name="struts.objectFactory" value="spring"></constant> <!-- 支持页面使用静态方法和属性 --> <constant name="struts.ognl.allowStaticMethodAccess" value="true"></constant> <!-- 跳转到登录页 --> <package name="CommonPackage" extends="struts-default" namespace="/common"> <action name="toLoginPage"> <result>/Web/login/page/login.jsp</result> </action> </package> <!-- 指定其它的配置文件路径 --> <include file="../conf/common/struts2/interceptor-common-exception.xml"></include> </struts>
//取得HttpServletRequest对象 HttpServletRequest request = ServletActionContext. getRequest(); //取得HttpSession对象 HttpSession session = ServletActionContext. getRequest().getSession();
public class BaseAction extends ActionSupport implements ServletRequestAware, ServletContextAware,ServletResponseAware{ //配置序列化标志 private static final long serialVersionUID = 1L; //request,response,application,session作用域 protected HttpServletRequest request ; protected ServletContext application; protected HttpServletResponse response; protected HttpSession session; /** * 依赖servlet容器的IOC设置-start */ @Override public void setServletRequest(HttpServletRequest arg0) { this.request = arg0; session = this.request.getSession(); } @Override public void setServletContext(ServletContext arg0) { this.application = arg0; } @Override public void setServletResponse(HttpServletResponse arg0) { this.response = arg0; } /** * 依赖servlet容器的IOC设置-end */ //重写execute方法 @Override public String execute() throws Exception { request.setAttribute("requestKey", "requestValue"); session.setAttribute("sessionKey", "sessionValue"); application.setAttribute("applicationKey", "applicationValue"); return SUCCESS; } }}
OGNL是一个功能强大的EL,通过它简单一致的表达式语法,可以存取对象的任意属性,调用对象的方法,遍历整个对象的结构图,实现字段类型转化等功能。它使用相同的表达式去存取对象的属性。因为太过复杂,这里就简单介绍几个常用的。
假设demo1.jsp页面有姓名name,年龄age这两个表单数据,它们被提交到action(action可以访问数据库保存或者比对或者更新),然后action再转发到demo2.jsp页面显示刚才输入的数据。
这是一个典型的B/S案例,我们可以将表单数据封装到一个JavaBean中(有的人喜欢叫pojo,还有的叫vo,还有的叫formbean),到demo2.jsp的时候通过ognl表达式取出来。
【demo1.jsp】
<%@ page contentType="text/html; charset=UTF-8" language="java" import="java.util.*" pageEncoding="UTF-8"%> <!--添加struts2标签的引用--> <%@ taglib prefix="s" uri="/struts-tags" %> <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"> <html> <head> </head> <body> <s:form action="doSaveAction.action" method="post"> <s:textfield name="user.name">姓名:</s:textfield> <s:textfield name="user.age">年龄:</s:textfield> <s:submit value="提交"></s:submit> </s:form> </body> </html>
【User.java】
public class User{ private String name; private String age; ...get、set方法... }【 SaveAction.java 】
public class SaveAction extends ActionSupport{ private static final long serialVersionUID = 1L; //将User这个javabean创建set方法便能获得从jsp请求过来的user对象数据 //同时user对象会放在值栈中,在demo1.jsp--SaveAction--demo2.jsp这个生命周期都存在 private User user; public String doSave(){ return SUCCESS; } public User getUser() { return user; } public void setUser(User user) { this.user = user; } }【 demo2.jsp】
<s:property value="user.name"/> <s:property value="user.age"/>之所以命名为OGNL,就是因为它处理对象很给力,struts能够将对象层层解析,把各个对象的关系以图的样式展示出来。比如user.name,之所以能找到这个对象,就是因为OGNL会先找user对象,然后再找user对象里的name对象。假设User这个类还包含了名为Role的javabean的实例,Role里面包含字段roleName,我们要找到roleName就可以直接写<s:property value="user.role.roleName">,OGNL通过对象逐级导航找到子对象。
当然OGNL也可以像jsp一样调用后台代码,不过我发现在jsp调用后台代码的效率很低,所以不是很推荐。
ognl调用静态方法的写法如下:@类全名@静态方法名,这句话等同于java代码:类全名.静态方法名
运行静态方法:<s:property value="@com.bless.ssh2.action.OgnlAction@staticMethod()"/><br/> 运行值栈对象中的方法:<s:property value="user.method()"/><br/> 调用JDK中的静态方法:<s:property value="@java.lang.Math@floor(44.56)"/><br/>需要注意的是,struts2.1.*默认是不允许调用静态方法的,你需要在struts.xml配置这样一句话:
<constant name="struts.ognl.allowStaticMethodAccess" value="true"/>
下面这个是早前自己写的Struts2标签库的例子,内容肯定不全:
<body> <!-- "%{}"表示支持ognl表达式,"#u"表示从值栈的actioncontex中查找值 --> <s:form action="doTags.action"> <h3>常用标签库</h3> <table> <tr> <td>编号:</td> <!-- 直接获取后台的值 --> <td><s:property value="tagBean.id"></s:property></td> </tr> <tr> <td>姓名:</td> <!-- 文本框 --> <td><s:textfield name="tagBean.name"></s:textfield></td> </tr> <tr> <td>密码:</td> <!-- 密码框 --> <td><s:password name="tagBean.password"></s:password></td> </tr> <tr> <td>生日:</td> <td> <!-- 在文本框中格式化时间 --> <s:textfield name="tagBean.birthday"> <s:param name="value"> <s:date name="tagBean.birthday" format="yyyy-MM-dd hh:MM:ss" /> </s:param> </s:textfield> </td> <td>生日:</td> <!-- 时间标签,可以格式化滴 --> <td><s:date name="tagBean.birthday" format="yyyy年MM月dd日"></s:date></td> </tr> <tr> <td>性别:</td> <td><s:radio name="tagBean.sex" list="#request.lstSex" listKey="sexId" listValue="sexValue"></s:radio></td> </tr> <tr> <td>城市:</td> <!-- listKey:下拉列表被选中后的真实值,listValue:下拉列表被选中后的显示值,headerKey:默认值,headerValue:默认显示值,emptyOption:是否在默认值和实际值下拉列表之间加一个空行 --> <td><s:select name="tagBean.city" list="#request.lstCity" listKey="cityId" listValue="cityValue" headerKey="0" headerValue="--请选择--" emptyOption="true"></s:select></td> </tr> <tr> <!-- 此标签等同于:struts2.tags.ListAddress la = new struts2.tags.ListAddress(); --> <!-- 此标签需要启用struts2的一个配置常量:<constant name="struts.enable.DynamicMethodInvocation" value="true" />,通常不推荐使用此法,在这只是做个例子 --> <s:bean name="struts2.tags.ListAddress" id="la"></s:bean> <td>地区:</td> <!-- s:checkboxlist标签支持多个checkbox, s:checkbox只支持单个checkbox--> <td><s:checkboxlist name="tagBean.address" list="#la.lstAddress" listKey="addressId" listValue="addressValue"></s:checkboxlist></td> </tr> </table> <!-- hidden标签存放隐藏值 --> <s:hidden></s:hidden> <s:submit value="确定"></s:submit> <s:reset value="重置"></s:reset> <hr/> <h3>OGNL</h3> <table> <!-- #符号: 获取值栈中的作用域 [parameters]路径中传参,[request][session][application]这几个不用说,[attr]表示从parameters到application依次找--> <!-- %{}符号:可以获取值栈中的对象,可以使用ognl表达式 --> <!-- 如果需要用到直接调用静态方法,需要启用struts2的一个配置常量:<constant name="struts.ognl.allowStaticMethodAccess" value="true"/> --> <tr> <td>直接调用静态方法:</td> <td><s:property value="@struts2.tags.Ognl@staticMethod()"/></td> </tr> <tr> <td>直接调用静态属性:</td> <td><s:property value="@struts2.tags.Ognl@STATIC"/></td> </tr> <tr> <td>调用值栈对象中的普通方法:</td> <td><s:property value="tagBean.normalMethod()"/></td> </tr> <tr> <td>调用JDK的静态方法(带参):</td> <td><s:property value="@java.lang.Math@floor(11.11)"/></td> <td>吓死你:</td> <!-- 如果第一个@后不带任何类,就默认表示java.lang.Math这个工具类,其它类不能这样写 --> <td><s:property value="@@floor(22.22)"/></td> </tr> <tr> <td>new一个对象:</td> <td><s:property value="new struts2.tags.Ognl('构造参数').result"></s:property></td> </tr> <tr> <td>#获取request数据:</td> <td><s:property value="#request.lstCity"></s:property></td> </tr> <tr> <td>#获取session数据:</td> <td><s:property value="#session.sess"></s:property></td> </tr> <tr> <td>下标-获取List集合第一条数据:</td> <td><s:property value="#request.lstCity[0]"></s:property></td> <!-- 注:Set集合是无序的所以不能通过下标获取数据;Map集合是键值对的,获取的时候应该根据key找value:map['key'] --> </tr> <tr> <td>获取List数据总条数:</td> <td><s:property value="#request.lstCity.size"></s:property></td> </tr> <tr> <td>投影技术获取List中指定对象数组:</td> <td><s:property value="#request.lstCity.{cityValue}"></s:property></td> </tr> <tr> <td>投影技术+下标获取一个值:</td> <td><s:property value="#request.lstCity.{cityValue}[0]"></s:property></td> </tr> <tr> <td>选择器:</td> <td><s:property value="#request.lstCity.{#this.cityId<=2}"></s:property></td> </tr> <tr> <td>选择器(?)+投影</td> <!-- ?表示获得所有满足条件的对象 --> <td><s:property value="#request.lstCity.{?#this.cityId<=2}.{cityId}"></s:property></td> </tr> <tr> <td>选择器(^)+投影</td> <!-- ^表示取得第一个满足条件的元素 --> <td><s:property value="#request.lstCity.{^#this.cityId<=2}.{cityId}"></s:property></td> </tr> <tr> <td>选择器($)+投影</td> <!-- $表示取得最后一个满足条件的元素 --> <td><s:property value="#request.lstCity.{$#this.cityId<=2}.{cityId}"></s:property></td> </tr> <tr> <td>Top语法:</td> <!-- .top取栈中的对象 ,[n]表示获取栈中第几个对象(从0开始)--> <td><s:property value="[0].top.locale"></s:property></td> </tr> <tr> <td>[n]语法:</td> <!-- [n]表示获取栈中第几个对象(从0开始)--> <td><s:property value="[0].locale"></s:property></td> </tr> </table> <hr/> <h3>逻辑,迭代标签</h3> <!-- test代码块中支持其它数据类型,如果是int的可以这样:test="#request.if==1" --> --if标签-- <s:if test="#request.if=='if'"> 这里是if模块 </s:if><s:elseif test="#request.if=='elseif'"> 这里是elseif模块 </s:elseif><s:else> 这里是else模块 </s:else> <br/><br/> --循环标签①-- <!-- first表示起始值,last表示结束值 iterator表示内部循环代码--> <s:bean name="org.apache.struts2.util.Counter" id="counter"> <s:param name="first" value="3" /> <s:param name="last" value="10" /> <s:iterator> <s:property/> </s:iterator> </s:bean> <br/><br/> --循环标签②-- <s:iterator begin="1" end="10"> <s:property/> </s:iterator> <br/><br/> --迭代list-- <!-- 等同于for(int i=1;i<=lstCity.size;i++) 标签同样从0开始计数 --> <s:iterator value="#request.lstCity" status="st" var="value" begin="1" end="#request.lstCity.length"> [<s:property value="%{#attr.st.index+1}"/>] <s:property value="%{#attr.value.cityValue}"/> <!-- 或者使用EL表达式取值:${value.cityId} --> </s:iterator> </s:form> <s:debug></s:debug> </body>
<tr> <td>生日:</td> <td> <!-- 在文本框中格式化时间 --> <s:textfield name="tagBean.birthday"> <s:param name="value"> <s:date name="tagBean.birthday" format="yyyy-MM-dd hh:MM:ss" /> </s:param> </s:textfield> </td> <td>生日:</td> <!-- 时间标签,可以格式化滴 --> <td><s:date name="tagBean.birthday" format="yyyy年MM月dd日"></s:date></td> </tr>
<struts> <package name="bless-default" extends="struts-default" > <interceptors> <!--定义一个名为privilege的拦截器--> <interceptor class="....ExceptionHandler" name="exceptionInterceptor"/> <!--①定义一个包含异常处理的拦截器栈--> <interceptor-stack name="exceptionStack"> <!-- 默认的拦截器 --> <interceptor-ref name="defaultStack"/> <!-- 异常拦截器 --> <interceptor-ref name="exceptionInterceptor"/> </interceptor-stack> <!-- ③需要用到chain跳转的拦截器栈 --> <interceptor-stack name="chainStack"> <interceptor-ref name="chain"/> <interceptor-ref name="basicStack"/> <interceptor-ref name="exceptionInterceptor"/> </interceptor-stack> </interceptors> <!-- 默认运行的拦截器栈 --> <default-interceptor-ref name="exceptionStack" /> <global-results> <result name="error" >/Web/common/page/error.jsp</result> </global-results> <global-exception-mappings> <exception-mapping result="error" exception="java.lang.Exception"></exception-mapping> </global-exception-mappings> </package> </struts>
public class ExceptionHandler extends AbstractInterceptor { private static final long serialVersionUID = 1L; public String intercept(ActionInvocation actioninvocation) { String result = null; // Action的返回值 try { // 运行被拦截的Action,期间如果发生异常会被catch住 result = actioninvocation.invoke(); return result; } catch (Exception e) { /** * 处理异常 */ String errorCode = null; // ospms的异常 int errorId = -1; // 达运的异常 if (e instanceof FunctionException) { // OSPMS异常 FunctionException fe = (FunctionException) e; errorCode = fe.getErrorCode(); fe.printStackTrace(); }........... /** * 发送错误消息到页面 */ ........... /** * log4j记录日志 */ ........... return "error"; }// ...end of catch } }