Blog: http://my.csdn.net/peng_hao1988
版本总览:http://blog.csdn.net/peng_hao1988/article/details/9026897
实现步骤(在上一版本上修改)
一、①.web.xml的配置修改如下:
<?xml version="1.0" encoding="UTF-8"?>
<web-app xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns="http://java.sun.com/xml/ns/javaee" xmlns:web="http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd" xsi:schemaLocation="http://java.sun.com/xml/ns/javaee http://java.sun.com/xml/ns/javaee/web-app_3_0.xsd" version="3.0">
<display-name></display-name>
<welcome-file-list>
<welcome-file>index.jsp</welcome-file>
</welcome-file-list>
<listener>
<description>Spring core configuration</description>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
<filter>
<filter-name>struts2</filter-name>
<filter-class>
org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter
</filter-class>
<init-param>
<param-name>config</param-name>
<!-- 示例中将struts2的配置文件struts.xml文件放到了WEB-INF根目录下 -->
<param-value>struts-default.xml,struts-plugin.xml,../struts.xml</param-value>
</init-param>
</filter>
<filter-mapping>
<filter-name>struts2</filter-name>
<url-pattern>*.action</url-pattern>
<!-- 这里很关键,如果不配置,从JSP页面就不能通过JSP标签转向到action -->
<dispatcher>REQUEST</dispatcher>
<dispatcher>FORWARD</dispatcher>
</filter-mapping>
</web-app>
注意上述配置,dispatcher中添加REQUEST和FORWARD两个配置,为了启用request和forward请求过滤,这样就可以使用request转发到action。②.spring配置文件中使用properties文件配置数据源(示例的db-config.properties文件放在WEB-INF下)代码如下:
<!-- 配置spring资源文件 -->
<bean id="propertyConfigurer" class="org.springframework.beans.factory.config.PropertyPlaceholderConfigurer">
<property name="locations">
<list>
<value>/WEB-INF/db-config.properties</value>
</list>
</property>
</bean>
<!-- 数据源 -->
<bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource">
<property name="driverClassName" value="${driverClass}">
</property>
<property name="url" value="${url}">
</property>
<property name="username" value="${username}"></property>
<property name="password" value="${password}"></property>
</bean>
③.db-config.properties文件的内容如下:## database driver class
driverClass=com.mysql.jdbc.Driver
## connection url
url=jdbc:mysql://localhost:3306/study
## user name and password for database
username=root
password=admin
## hibernate dialect
hibernate.dialect=org.hibernate.dialect.MySQL5Dialect
## create or update database
hibernate.hbm2ddl.auto=update
## whether show SQL
hibernate.show_sql=false
hibernate.format_sql=true
欢迎页面index.jsp页面通过转发或重定向到login.jsp,代码如下:
<s:form action="userManagerAct" namespace="/" method="post" theme="simple">
<table>
<tr>
<td>username</td><td><s:textfield name="user.uname"/></td>
<td>password</td><td><s:password name="user.password"/></td>
<td colspan="2">
<s:submit value="submit" />
<s:reset value="reset"/>
</td>
</tr>
</table>
</s:form>
因为login.jsp中使用了struts标签,所以需要重action中进入页面,否则会报错!如果不想通过action到页面,可以在login.jsp中使用普通html标签,然后用EL表达式辅助回显。二、从欢迎页面index.jsp到login.jsp有多种方式,body中的代码如下:
<%--(1). 转发一个HTML文件,JSP文件,或者是一个程序段.
<jsp:forward page="/userManagerAct.action"></jsp:forward>
--%>
<%--(2). 通过request转发
<%
request.getRequestDispatcher("/userManagerAct.action").forward(request, response);
%>
--%>
<%--(3). 重定向 --%>
<%
response.sendRedirect("userManagerAct.action");
%>
上述三种方式中(1)、(2)为请求转发,(3)为重定向,前两种方式能成功的前提条件是web.xml文件中将request好forward配置到了struts2的filter-mapping中了,否则会报404错误!三、在通过struts标签请求action是注意:
不要在action名称后面添加后缀(本例中后缀为*.action),因为使用struts标签会自己处理后缀,如果添加了后缀,后台会抛出一个警告
No configuration found for the specified action: 'xxxx.action' in namespace: '/'. Form action defaulting to 'action' attribute's literal value.
网上一部分资料中说上述警告是由于没有添加namespace导致的,但是本人测试时发现不给action添加后缀,无论有无namespace都没有发现警告,若添加上后缀就会出现警告。具体什么原因就不详究了,大家在开发是注意不要给struts2标签请求的action名称后加后缀,form中最好添加上namespace。非struts2标签的请求(普通的html标签、js、以及如index.jsp中)必须添加后缀(如果在web.xml中配置了后缀的话)。
四、拦截器(Interceptor)配置,实现用户登录状态验证,操作权限验证。如示例中登录状态验证和删除权限验证配置如下:
<package name="myApp" extends="struts-default">
<interceptors>
<interceptor name="loginInterceptor" class="com.hzboy.common.LoginInterceptor"/>
<interceptor name="delInterceptor" class="com.hzboy.common.DelInterceptor"/>
<!-- 自定义拦截器栈,扩展原有拦截器特性 -->
<interceptor-stack name="MyInterceptors">
<interceptor-ref name="defaultStack"/>
<interceptor-ref name="loginInterceptor"/>
</interceptor-stack>
<!-- 定义删除信息拦截器栈 -->
<interceptor-stack name="delInterceptors">
<interceptor-ref name="MyInterceptors"/>
<interceptor-ref name="delInterceptor"/>
</interceptor-stack>
</interceptors>
<!-- 修改默认拦截器,在默认拦截器的基础之上添加了登录验证 -->
<default-interceptor-ref name="MyInterceptors"/>
<global-results>
<result name="login">/login.jsp</result>
<result name="query">/WEB-INF/jsp/success.jsp</result>
</global-results>
<action name="userManagerAct" class="userAct" method="doLogin">
<!-- 因为修改了默认拦截器,添加了登录验证,所以不能再使用默认的拦截器,登录请求不需要验证,故使用struts默认的拦截器 -->
<interceptor-ref name="defaultStack"/>
</action>
<action name="query" class="userAct" method="doQuery" />
<action name="delete" class="userAct" method="doDelete">
<interceptor-ref name="delInterceptors"/>
<result name="delMsg">/WEB-INF/jsp/message.jsp</result>
</action>
<action name="edit" class="userAct" method="doEdit">
<result name="editUser">/WEB-INF/jsp/editUser.jsp</result>
</action>
<action name="add" class="userAct" method="doAdd">
<result name="addUser">/WEB-INF/jsp/addUser.jsp</result>
</action>
</package>
①. 自定义拦截器:在package中定义,并且可以通过package的继承来被其他的子package引用,可以将多个拦截器在interceptor-stack中引用组成一个拦截器栈,拦截器栈会按照引用的顺序依次执行。action中添加拦截器或拦截器栈使用interceptor-ref来实现。如果想让某个拦截器或拦截器栈作为默认拦截器(即替换原有的defaultStack)可以使用default-interceptor-ref来指定。默认的拦截器会拦截前包和子包中的所有action请求,所以除非当前包下所有action都需要某个拦截操作,否则没有必要指定default-interceptor-ref、如果指定一定要,那应该指定一个拦截器栈,而且该栈中引用了defaultStack。
注意:当action应用一个拦截或拦截器栈(未引用defaultStack)时,struts2默认的拦截器就会失效,会导致部分功能无法使用,如struts2标签的回显。
②.定义拦截器:定义一个java类继承com.opensymphony.xwork2.interceptor.AbstractInterceptor类并实现其intercept方法,以LoginInterceptor为例,代码如下:
public class LoginInterceptor extends AbstractInterceptor{
private static final long serialVersionUID = 1L;
private Logger logger = Logger.getLogger(LoginInterceptor.class);
@Override
public String intercept(ActionInvocation ai) throws Exception {
logger.info("LoginInterceptor is runing....");
Object user = ai.getInvocationContext().getSession().get(Global.SESSION_USER_INFO);
if(user != null)
return ai.invoke();
else
return Global.RESULT_LOGIN;
}
}
通过参数ActionInvocation类的invoke()方法返回到控制器继续执行后面的拦截器,如果没有就直接转发到请求的action,可以返回一个字符串到指定页面或action,该字符串应该是被拦截的action中的result的name值或是global-results中某个result的name值,否则就报404。详细的struts2拦截器配置及介绍参见网友博文:
http://blog.csdn.net/qjyong/article/details/1824607 。
五、属性验证、Validate
①.通常从web页面提交到action的数据需要做一些有效性验证,常见的就是在action中直接验证属性值,如下实例中的添加用户代码:
/**
* @description 人员信息添加
* @return
* @author porter
* @created 2013-6-2 下午6:13:56
*/
public String doAdd(){
logger.info("doAdd is runing...");
String result = Global.RESULT_QUERY;
try {
String param = (String)this.requestGetParam("param");
if(param != null && Integer.parseInt(param) > 0){
if(validatation()){
this.user.setId(0);
userService.addUser(this.user);
result = doQuery();
}else
result = Global.RESULT_ADD_USER;
}else
result = Global.RESULT_ADD_USER;
} catch (Exception e) {
logger.error(e);
}
return result;
}
/**
* @description 验证添加的人员信息
* @return
* @author porter
* @created 2013-6-2 下午6:14:24
*/
public boolean validatation() {
if(this.user.getUname() == null || this.user.getUname().isEmpty()){
this.addFieldError("user.uname", "Username cant not be null!");
return false;
}else if(this.user.getPassword() == null || this.user.getPassword().isEmpty()){
this.addFieldError("user.uname", "password cant not be null!");
return false;
}
return true;
}
添加用户信息是用户名和密码不能为空,如果不符合要就就踢回到web页面并提示用户,提示信息可以使用addActionError,addFieldError,addActionMessage。 三种消息的用法介绍(引用)
②.还有一种验证方式:使用xml描述来验证action中的属性即validate框架。该框架需要额外添加xml配置,比较麻烦,实际开发中使用较少,详细可参见网友博文
六、struts-json,通过ajax和json-plugin来实现数据的后台验证,如实例中添加用户信息界面,当输入完成用户名后使用ajax去有后台验证用户名是否以被占用。实例代码:
<!-- 配置JSON响应方法 -->
<package name="my-json" extends="json-default">
<action name="checkUName" class="userAct" method="checkUName">
<result type="json" name="success">
<!-- 只返回user.name的值,没有key -->
<param name="root">user.uname</param>
<!-- 需要去除的属性
<param name="excludeProperties">userService,users,searchText</param>
-->
<!-- 需要包含的属性
<param name="includeProperties">user\.uname</param>
-->
</result>
</action>
</package>
result中type值的取值范围和ajax数据格式基本一致,有json,xml、text等,name值可以随意,但是action执行完后返回的结果必须和这里的name值匹配。param参数有一下几种:
1.root参数:从返回结果中根据ognl表达式取出你需要输出的结果
2.excludeNullProperties 参数:表示是否去掉空值, 默认值是false,如果设置为true会自动将为空的值过滤,只输出不为空的值。
3.ignoreHierarchy 参数:表示是否忽略等级,也就是继承关系,比如:TestAction继承于BaseAction,那么TestAction中返回的json字符串默认是不会包含父类BaseAction的属性值,ignoreHierarchy值默认为true,设置为false后会将父类和子类的属性一起返回
4.includeProperties 参数:输出结果中需要包含的属性值,这里正则表达式和属性名匹配,可以用“,”分割填充多个正则表达式。
5.excludeProperties 参数:输出结果需要剔除的属性值,也支持正则表达式匹配属性名,可以用“,”分割填充多个正则表达式。
页面js中的代码如下:
$(function(){
//给用户名输入框设置焦点事件,当失去焦点是触发事件,判断用户输入的用户名是否合法!
$("input[name='user.uname']").blur(function(){
var uname = $(this).val();
if(uname == ""){
$("#chkMsg").css({"color":"red"});
$("#chkMsg").text("用户名不能为空!");
return;
}
$.ajax({
type:"post",
url:"checkUName.action",
data:{"uname":uname},
dataType:"json",
success:function(data){
if(data == null){
$("#chkMsg").css({"color":"#00cc00"});
$("#chkMsg").text("该用户名可以使用!");
}else{
$("#chkMsg").css({"color":"red"});
$("#chkMsg").text("该用户名已被使用!");
}
}
});
});
});
注意:使用struts-json来完成数据交换需要添加struts2-json-plugin-2.x,x.jar到工程的classpath中。
示例代码:http://download.csdn.net/detail/peng_hao1988/5542325
》》》》》》》》》》》》》》》》转载请注明出处《《《《《《《《《《《《《《《《