Struts 2 实现Action的几种方式
Action用于处理用户的请求,因此也被称为业务控制器。每个Action类就是一个工作单元,Struts2框架负责将用户的请求与相应的Action匹配,如果匹配成功,则调用该Action类对用户请求进行处理,而匹配规则需要在Struts2的配置文件中声明。
Struts2框架下实现Action类有以下三种方式:
- 普通的POJO类,该类通常包含一个无参数的execute()方法,返回值为字符串类型。
- 实现Action接口
- 继承ActionSupport类
POJO实现方式
以用户登录为例,创建LoginAction类。
packagecom.qst.chapter03.action; publicclassLoginAction{ /*用户名*/ privateStringuserName; /*密码*/ privateStringpassword; publicStringgetUserName(){ returnuserName; } publicvoidsetUserName(StringuserName){ this.userName=userName; } publicStringgetPassword(){ returnpassword; } publicvoidsetPassword(Stringpassword){ this.password=password; } /** *调用业务逻辑方法,控制业务流程 */ publicStringexecute(){ System.out.println("----登录的用户信息-----"); System.out.println("用户名:"+userName); System.out.println("密码:"+password); if(userName.startsWith("qst")&&password.length()>=6){ //返回成功页面 return"ok"; }else{ //返回失败页面 return"error"; } } }
登录页面:
<%@pagelanguage="java"contentType="text/html;charset=UTF-8" pageEncoding="UTF-8"%>用户登录
用户名 密码
错误页面:
<%@pagelanguage="java"contentType="text/html;charset=UTF-8" pageEncoding="UTF-8"%>错误页面 登录失败!
成功页面:
<%@pagelanguage="java"contentType="text/html;charset=UTF-8" pageEncoding="UTF-8"%>显示用户信息 登录成功!欢迎用户${param.userName}!
struts.xml:
/ok.jsp /error.jsp
这样就以POJO方式实现了一个Action,当单击“登录按钮时”,表单中的数据会提交给login.action,Struts2框架将自动调用LoginAction的setter方法将请求参数值封装到对应的属性中,并执行execute()方法。
实现Action接口方式
为了让Action类更规范,使各个开发人员编写的execute()方法返回的字符串风格是一致的,Struts2提供一个Action接口,该接口定义了Acitoin处理类应该实现的通用规范:
publicabstractinterfaceAction{ publicstaticfinaljava.lang.StringSUCCESS="success"; publicstaticfinaljava.lang.StringNONE="none"; publicstaticfinaljava.lang.StringERROR="error"; publicstaticfinaljava.lang.StringINPUT="input"; publicstaticfinaljava.lang.StringLOGIN="login"; publicStringexecute()throwsException; }
下面代码使用Action接口来创建Action类:
packagecom.qst.chapter03.action; importcom.opensymphony.xwork2.Action; publicclassLoginAction2implementsAction{ /*用户名*/ privateStringuserName; /*密码*/ privateStringpassword; publicStringgetUserName(){ returnuserName; } publicvoidsetUserName(StringuserName){ this.userName=userName; } publicStringgetPassword(){ returnpassword; } publicvoidsetPassword(Stringpassword){ this.password=password; } /** *调用业务逻辑方法,控制业务流程 */ publicStringexecute(){ System.out.println("----登录的用户信息-----"); System.out.println("用户名:"+userName); System.out.println("密码:"+password); if(userName.startsWith("qst")&&password.length()>=6){ //返回成功页面 returnSUCCESS; }else{ //返回失败页面 returnERROR; } } }
struts.xml:
/ok.jsp /error.jsp
继承ActionSupport类方式
Struts2框架为Action接口提供了一个实现类ActionSupport,该类提供了许多默认方法,写Action类时继承ActionSupport类会大大简化Action的开发。ActionSupport类是Struts2默认的Action处理类,如果配置Action类时没有指定class属性,系统自动默认使用ActionSupport类作为Action的处理类。
下面代码通过继承ActionSupport类来创建Action类,并重写validate()验证方法:
packagecom.qst.chapter03.action; importcom.opensymphony.xwork2.ActionSupport; publicclassLoginAction3extendsActionSupport{ /*用户名*/ privateStringuserName; /*密码*/ privateStringpassword; publicStringgetUserName(){ returnuserName; } publicvoidsetUserName(StringuserName){ this.userName=userName; } publicStringgetPassword(){ returnpassword; } publicvoidsetPassword(Stringpassword){ this.password=password; } /** *调用业务逻辑方法,控制业务流程 */ publicStringexecute(){ System.out.println("----登录的用户信息-----"); System.out.println("用户名:"+userName); System.out.println("密码:"+password); if(userName.startsWith("qst")&&password.length()>=6){ //返回成功页面 returnSUCCESS; }else{ //返回失败页面 returnERROR; } } //重写validate()方法 publicvoidvalidate(){ //简单验证用户输入 if(this.userName==null||this.userName.equals("")){ //将错误信息写入到Action类的FieldErrors中 //此时Struts2框架自动返回INPUT视图 this.addFieldError("userName","用户名不能为空!"); System.out.println("用户名为空!"); } if(this.password==null||this.password.length()<6){ this.addFieldError("password","密码不能为空且密码长度不能小于6"); System.out.println("密码不能为空且密码长度不能小于6!"); } } }
上述代码增加了一个对表单提交的数据进行验证的validate()方法,该方法会在执行execute()方法之前运行,如果发现表单提交数据不符合要求,则执行addFieldError()方法,将错误信息写入Action类的字段错误列表FieldErrors中,并且将自动返回到INPUT输入视图,让用户重新输入表单并提交。
在struts.xml配置文件中增加INPUT输入视图:
/ok.jsp /error.jsp /login.jsp
当表单提交的数据验证不通过时,则会返回到输入页面,程序会依然“停留”(看起来是这样,但其实是一个新的输入页面)在输入页面login.jsp。
访问ActionContext
相对于Struts1,Struts2的一个重要改进是使Action不再和任何ServletAPI耦合,但有些时候Action类不访问ServletAPI是不能实现业务逻辑的(例如跟踪HTTPSession的状态)。此时Action就需要访问ServletAPI中的HttpSession。
Struts2提供了一种更加轻松的方式来访问ServletAPI。在Struts2框架中,Action可以通过ActionContext类来访问ServletAPI,ActionContext提供了读写ServletAPI中的HttpServletRequest、HttpSession和ServletContext中数据的方法。
常用方法如下表所示:
方法
功能描述
Objectget(Objectkey)
获取属性值,与HttpSevletRequest的getAttribute(Stringname)类似
MapgetApplication()
返回一个Map对象,该对象模拟了Web应用对应的ServletContext对象
staticActionContextgetContext()
静态方法,用于获取系统的ActionContext对象
MapgetParameters()
获取所有的请求参数,类似于调用HttpSevletRequest对象的getParameterMap()方法
MapgetSession()
返回一个Map对象,该对象模拟了HttpSession实例
voidsetApplication(Mapapplication)
直接传入一个Map对象,并将该Map对象中的键/值对转换成application的属性名和属性值
voidsetSession(Mapsession)
直接传入一个Map对象,并将该Map对象中的键/值对转换成session的属性名和属性值
下面代码演示Action访问ActionContext的使用
packagecom.qst.chapter03.action; importcom.opensymphony.xwork2.ActionContext; importcom.opensymphony.xwork2.ActionSupport; publicclassClickNumActionextendsActionSupport{ publicStringexecute(){ //获取ActionContext对象,通过该对象访问ServletAPI ActionContextctx=ActionContext.getContext(); //获取ServletContext里的num属性 Integernum=(Integer)ctx.getApplication().get("num"); //如果num属性为null,设置num属性为1 if(num==null){ num=1; }else{ //将num加1 num++; } //将加1后的num值保存在application中 ctx.getApplication().put("num",num); returnSUCCESS; } }
上面代码先使用ActionContext.getContext()静态方法获取系统的ActionContext对象,再调用ActionContext对象的getApplication()方法获取ServletContext对应的Map对象,然后调用get()/put()方法进行数据的读/写操作。最后将num值保存到ServletContext中。
编写clickNum.jsp页面,对application进行访问:
<%@pagelanguage="java"contentType="text/html;charset=UTF-8" pageEncoding="UTF-8"%>点击次数统计 点击按钮,已点击了 ${emptyapplicationScope.num?0:applicationScope.num}次
上述代码创建了一个表单,表单的action属性值为clicknum,action,当单击“提交”按钮时会将表单提交给CounterAction进行处理。
在struts.xml配置文件中增加CounterAction的配置:
/ok.jsp /error.jsp /login.jsp /clickNum.jsp
访问ServletAPI
虽然Struts2提供了ActionContext来直接访问ServletAPI。但有时你需要直接访问SevletAPI。为了在Action中能够直接访问ServletAPI,Struts2还提供了一系列的其他接口,通过实现这些接口,Action可以直接访问ServletAPi。
Struts2框架提供的访问ServletAPI的接口如下表所示:
方法 | 功能描述 |
---|---|
ServletContextAware | 实现该接口的Action可以直接访问Web应用的SevletContext实例 |
ServletRequestAware | 实现该接口的Action可以直接访问用户请求的HttpServletRequest实例 |
ServletResponseAware | 实现该接口的Action可以直接访问服务器响应的HttpSevletResponse实例 |
下面代码以实现SevletRequestAware接口为例,通过获取HttpSession,将登陆成功的用户名保存到Session中:
packagecom.qst.chapter03.action; importjavax.servlet.http.HttpServletRequest; importjavax.servlet.http.HttpSession; importorg.apache.struts2.interceptor.ServletRequestAware; importcom.opensymphony.xwork2.ActionSupport; publicclassLoginAction4extendsActionSupportimplementsServletRequestAware{ /*用户名*/ privateStringuserName; /*密码*/ privateStringpassword; publicStringgetUserName(){ returnuserName; } publicvoidsetUserName(StringuserName){ this.userName=userName; } publicStringgetPassword(){ returnpassword; } publicvoidsetPassword(Stringpassword){ this.password=password; } //声明request对象 privateHttpServletRequestrequest; //重写ServletRequestAware中的方法 publicvoidsetServletRequest(HttpServletRequestrequest){ this.request=request; } /** *调用业务逻辑方法,控制业务流程 */ publicStringexecute(){ System.out.println("----登录的用户信息-----"); System.out.println("用户名:"+userName); System.out.println("密码:"+password); if(userName.startsWith("qst")&&password.length()>=6){ //获得session对象 HttpSessionsession=request.getSession(); //将用户名存放到session中 session.setAttribute("CurUser",userName); //返回成功页面 returnSUCCESS; }else{ //返回失败页面 returnERROR; } } //重写validate()方法 publicvoidvalidate(){ //简单验证用户输入 if(this.userName==null||this.userName.equals("")){ //将错误信息写入到Action类的FieldErrors中 //此时Struts2框架自动返回INPUT视图 this.addFieldError("userName","用户名不能为空!"); System.out.println("用户名为空!"); } if(this.password==null||this.password.length()<6){ this.addFieldError("password","密码不能为空且密码长度不能小于6"); System.out.println("密码不能为空且密码长度不能小于6!"); } } }
上述代码定义的LoginAction4实现了SevletRequestAware接口,并且重写该接口中setServletRequest()方法,setServletRequest()方法的参数是HttpServletRequest对象,运行Web应用时,Struts2框架会自动将当前请求对象传入setServletRequest()方法,再将该请求对象赋给LoginAction4的request属性,这样在LoginAction4类的其他方法中就可以访问到request对象了。通过request对象可以获取HttpSession对象,并将当前用户信息保存到Session中。
将login.jsp页面中表单的action属性改成login4.action:
......
创建first.jsp显示用户信息:
<%@pagelanguage="java"contentType="text/html;charset=UTF-8" pageEncoding="UTF-8"%>显示用户信息 登录成功!欢迎用户${param.userName}
当前用户${session.CurUser}
下一页
再创建一个second.jsp:
<%@pagelanguage="java"contentType="text/html;charset=UTF-8" pageEncoding="UTF-8"%>显示用户信息 请求中的用户信息:${param.userName}
Session中的用户信息:${session.CurUser}
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持毛票票。