这里的主要内容是:
- 中文乱码问题
- 简单数据校验
- 访问web元素
一、中文乱码问题
相关代码(工程Struts2_1000_CharacterEncoding
)
struts.xml
/user_add_success.jsp
UserAction.java
package com.bjsxt.struts2.user.action;
import com.opensymphony.xwork2.ActionSupport;
public class UserAction extends ActionSupport {
private String name;
public String add(){
System.out.println("name=" + name);
return SUCCESS;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
index.jsp
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8" isELIgnored="false"%>
CharacterEncoding
使用action属性接收参数,测试中文问题
user_add_success.jsp
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8" isELIgnored="false"%>
CharacterEncoding
User Add Success!
说明:解决中文乱码问题很简单,只需要配置:
当然我们需要统一编码格式。
二、简单数据校验
相关代码(工程Struts2_1100_SimpleDataValiation
)
struts.xml
/user_add_success.jsp
/user_add_error.jsp
UserAction.java
package com.bjsxt.struts2.user.action;
import com.opensymphony.xwork2.ActionSupport;
public class UserAction extends ActionSupport {
private String name;
public String add(){
if(name == null || !name.equals("admin")){
this.addFieldError("name", "name is error");
this.addFieldError("name", "name is too long");
return ERROR;
}
return SUCCESS;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
}
index.jsp
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8" isELIgnored="false"%>
SimpleDataValiation
使用addFieldError方法和s:fieldError标签简单处理数据校验
添加用户
user_add_error.jsp
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8" isELIgnored="false"%>
<%@taglib uri="/struts-tags" prefix="s" %>
SimpleDataValiation
User Add Error!
user_add_success.jsp
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8" isELIgnored="false"%>
SimpleDataValiation
User Add Success!
说明:
- 1.这里我们在action中进行简单是数据校验,当校验出错时跳转到错误页面,否则跳转到成功页面。
- 2.使用标签:
会默认是simple主题,会给我们加上一些标签用于展现,但大多数情况下我们并不需要这些标签,我们只需要相关的值,所以一般不实用此标签。 - 3.在错误页面中我们使用标签
则会帮我们生成一个超链接,点击超链接可以得到很多相关信息,我们使用标签
可以得到value stack
和action stack
中的值。这里我们使用errrors
关键字找到name
属性的错误信息,这相当于一个map
,而name
错误信息又是一个数组,我们取得数组第一个错误信息。同时我们也可以看到一个属性值可以对应多个错误信息。 - 4.后面两个标签以后会经常用到。
三、访问web元素
在使用struts2之后我们不能直接得到相关的request
、session
、application
等对象,但是有时候我们确实需要从这些对象中取得相关的值。这里struts2提供了Map
类型的request
、session
、application
分别对应真实类型的HttpServletRequest
、HttpSession
、ServletContext
的引用。这是三个最常用的对象引用,其他的对象如HttpServletResponse
就不需要了,因为我们是通过result
返回给页面的。下面通过例子说明如何取得相应的真实对象。
相关代码(工程Struts2_1200_AccessWebElements
)
struts.xml
/user_login_success.jsp
index.jsp
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8" isELIgnored="false"%>
AccessWebElements
取得Map类型request、session、application和其对应的真实对象HttpServletRequest、
HttpSession、ServeltContext的引用:
- 前三者:依赖于容器
- 前三者:IOC(只用这种)
- 后三者:依赖于容器
- 后三者:IOC
user_login_success.jsp
<%@ page language="java" import="java.util.*" pageEncoding="UTF-8" isELIgnored="false"%>
<%@taglib uri="/struts-tags" prefix="s" %>
AccessWebElements
User Login Success!
| <%=request.getAttribute("r1") %>
| <%=session.getAttribute("s1") %>
| <%=application.getAttribute("a1") %>
LoginAction1.java
package com.bjsxt.struts2.user.action;
import java.util.Map;
import com.opensymphony.xwork2.ActionContext;
import com.opensymphony.xwork2.ActionSupport;
public class LoginAction1 extends ActionSupport {
private Map request;
private Map session ;
private Map application ;
//这里我们可以在构造方法取得相关对象,也可以在execute方法中取得
public LoginAction1(){
request = (Map) ActionContext.getContext().get("request");
session = (Map) ActionContext.getContext().getSession();
application = (Map) ActionContext.getContext().getApplication();
}
public String execute(){
request.put("r1", "r1");
session.put("s1", "s1");
application.put("a1", "a1");
return SUCCESS;
}
}
说明:
- 1.对于
LoginAction1.java
,我们使用第一个提交按钮进行提交。在struts2中我们使用ActionContext
取得相关对象,我们可以在构造函数中取得,也可以在execute
方法中取得。 - 2.注意这里取得的对象是一个
Map
类型,我们可以在debug
链接中看到。我们存取值都是通过Map的相关方法取得。 - 3.在页面中我们通过标签取得相关值:
| <%=request.getAttribute("r1") %>
```
这里提供了两种方法,后面一种是最原始的方法,前一种是struts2提供的方法,我们在```debug```链接中可以看到这样一句话:```These items are available using the #key notation```,也就是我们通过这种方式取得相关的值。
* 4.最后三种方式也可以取得相关的值,但是一般不用,这种方式就是struts2会帮我们在整个```action stack```中进行搜索,直到找到相关的值。但是容易出错,因为很可能有不同域的相同属性和值。
* 5.其实```ActionContext```不是单例,而是一个```ThreadLocal```对象。即在某个线程中是单例。
* 6.这是第一种方式,但是这种方式依赖于struts2,即依赖于struts2的容器```ActionContext```,下面我们看第二种方式,使用IoC方法注入。
**```LoginAction2.java```**
package com.bjsxt.struts2.user.action;
import java.util.Map;
import org.apache.struts2.interceptor.ApplicationAware;
import org.apache.struts2.interceptor.RequestAware;
import org.apache.struts2.interceptor.SessionAware;
import com.opensymphony.xwork2.ActionSupport;
public class LoginAction2 extends ActionSupport implements RequestAware, SessionAware, ApplicationAware{
private Map request;
private Map session;
private Map application;
public String execute(){
request.put("r1", "r1");
session.put("s1", "s1");
application.put("a1", "a1");
return SUCCESS;
}
public void setRequest(Map request) {
this.request = request;
}
public void setSession(Map session) {
this.session = session;
}
public void setApplication(Map application) {
this.application = application;
}
}
**说明:**
* 1.这里我们实现了三个接口```RequestAware, SessionAware, ApplicationAware```,同时实现了相应的```setter```方法,当struts2收到浏览器的请求后就会将相关的真实对象注入到我们的这三个对象中,之后我们就可以使用了。
* 2.这里使用第二个提交按钮进行提交。之前的方式是我们自己去容器中拿,而这种方式是让struts2帮我们注入进来。我们一般使用这种方式。下面看最后两种方式,只需要了解,一般不用。
**```LoginAction3.java```**
package com.bjsxt.struts2.user.action;
import javax.servlet.ServletContext;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpSession;
import org.apache.struts2.ServletActionContext;
import com.opensymphony.xwork2.ActionSupport;
public class LoginAction3 extends ActionSupport {
private HttpServletRequest request ;
private HttpSession session;
private ServletContext application;
public LoginAction3(){
request = ServletActionContext.getRequest();
session = request.getSession();
application = session.getServletContext();
}
public String execute(){
request.setAttribute("r1", "r1");
session.setAttribute("s1", "s1");
application.setAttribute("a1", "a1");
return SUCCESS;
}
}
**说明:**可以看到这是一种最原始的方式,极度依赖容器,和之前完全不实用框架的方式是一样的。
**```LoginAction4.java```**
package com.bjsxt.struts2.user.action;
import javax.servlet.ServletContext;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpSession;
import org.apache.struts2.interceptor.ServletRequestAware;
import com.opensymphony.xwork2.ActionSupport;
public class LoginAction4 extends ActionSupport implements ServletRequestAware{
private HttpServletRequest request ;
private HttpSession session;
private ServletContext application;
public String execute(){
request.setAttribute("r1", "r1");
session.setAttribute("s1", "s1");
application.setAttribute("a1", "a1");
return SUCCESS;
}
@Override
public void setServletRequest(HttpServletRequest request) {
this.request = request;
this.session = request.getSession();
this.application = session.getServletContext();
}
}
**说明:**这种方式和第二种方式比较类似,也是使用注入的方式取得相关对象。
###最后:
一般来说,我们一般只使用第二种方式,最后两种方式一般是在我们需要用到原始```request```等对象的一些特殊方法才用。同时一般我们也值需要取得```session```对象,而```request```和```application```对象一般都不需要取得。其中```request```中的相关值我们在```stack value```中就可以得到,而```application```全局变量我们可以使用其他方式进行替代。