Struts2学习笔记

Struts2,经典前端框架,学习笔记

[TOC]

Struts2框架引言

什么是框架(FrameWork)

软件开发过程中的半成品,解决软件开发中的通用问题,从而提高开发效率。
eg

  • 字符集编码
  • 收集数据
  • 手工类型转化
  • JDBC访问

Struts2框架的概念

典型的MVC框架,人为的把一个软件分为3个层次从而提高开发效率。
M(Model | 模型层)Service + DAO + Entity
V(View | 视图层) JSP(freemarker velocity)
C(Controller | 控制层) (Servlet)

MVC设计思想的优点

  • 解耦合,利于代码维护
  • 有利于分工,提高代码开发效率

现有的MVC控制层所存在的问题

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
Servlet {
//收集数据
//出现问题1:设置字符集编码的代码问题
String age = request.getParamter("age");
request.getParamter("password");

//问题2:手工进行类型转换
Interger.parseInt();

//调用业务
Service

//跳转页面
request.getRequestDispatcher("/a.jsp").forward(request, response);

redirect;

//问题3:把跳转路径写死在程序中,不利于代码的维护
response.sendredirect("/b.jsp");

Struts2的实战开发思路

Struts2代码:

1
2
3
4
5
6
7
8
9
10
Struts2 {
MyStruts2 implements Action {
public String execute() throws Exception {
//调用业务
//调用业务
//跳转页面
return "hibiscidai"
}
}
}

对于WEBAPPLICATION的配置文件web.xml要声明映射

1
2
3
4
5
6
7
8
9
<servlet>
<servlet-name>A</servlet-name>
<servlet-class>xxx.servlet</servlet-class>
</servlet>

<servlet-mapping>
<servlet-name>A</servlet-name>
<url-pattern>/A</url-pattern>
</servlet-mapping>

对于Struts.xml的配置

1
2
3
4
<action name="A" class="xxx.action">
<result name="hibiscidai">
</result>
</action>

第一个Struts2程序的开发

搭建开发环境

引入核心jar包

struts2-core-2.3.15.1.jar

引入第三方jar包

引入Struts.xml配置文件

配置Struts2核心过滤器

在web.xml中声明

1
2
3
4
5
6
7
8
<filter>
<filter-name>struts2</filter-name>
<filter-class>org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>struts2</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>

对于不同版本会有兼容问题,注意 filter-class 标签映射类

开发步骤

实现Action接口

配置文件配置

8254

ServletActionContext类的使用

servlet-api.jar 包中

1
2
3
HttpServletRequest request = ServletActionContext.getRequest();
HttpServletResponse response =ServletActionContext.getResponse();
HttpSession session = request.getSession();

Strtus2的另一种访问方式

直接在项目路径下输入action名字或者 xxx.action
eg:
http://localhost:8080/ProjectName/MyAction
http://localhost:8080/ProjectName/MyAction.action

Struts2的跳转(4种|重点)

Action跳转JSP

默认Forward跳转

1
2
3
4
5
<action name="FirstStruts" class="fancylab.hibiscidai.action.MyAction">
<result name="success" type="dispatcher">
/ok.jsp
</result>
</action>

Redirect跳转

1
2
3
4
5
<action name="FirstStruts" class="fancylab.hibiscidai.action.MyAction">
<result name="success" type="redirect">
/ok.jsp
</result>
</action>

Action跳Action

Forward跳转

1
2
3
4
5
6
7
8
9
10
<action name="A" class="fancylab.hibiscidai.action.AAction">
<result name="B" type="chain">
B
</result>
</action>
<action name="B" class="fancylab.hibiscidai.action.BAction">
<result name="success">
/ok.jsp
</result>
</action>

Redirect跳转

1
2
3
4
5
6
7
8
9
10
<action name="A" class="fancylab.hibiscidai.action.AAction">
<result name="B" type="redirectAction">
B
</result>
</action>
<action name="B" class="fancylab.hibiscidai.action.BAction">
<result name="success">
/ok.jsp
</result>
</action>

包 < PACKAGE >

使配置文件当中的配置信息模块化,便于配置信息的管理。

1
<package name="xxx" extends="struts-default">

命名空间 < NAMESPACE >

使用户的请求模块化,便于随后过滤器的使用。

web.xml配置

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
<servlet>
<servlet-name>A</servlet-name>
<servlet-class>fancylab.hibiscidai.action.AAction</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>A</servlet-name>
<url-pattern>/User/A</url-pattern>
</servlet-mapping>
<servlet>
<servlet-name>A</servlet-name>
<servlet-class>fancylab.hibiscidai.action.AAction</servlet-class>
</servlet>
<servlet-mapping>
<servlet-name>A</servlet-name>
<url-pattern>/User/B</url-pattern>
</servlet-mapping>
<filter>
/User/*
</filter>

访问方式:localhost:[port]/ProjectName/A

Struts2 包空间加入后
struts.xml配置

1
2
3
4
5
6
<package name="user" extends="strust-default" namespace="/First">
<action name="A" class="fancylab.hibiscidai.action.AAction">
</package>
<package name="Admin" extends="strust-default">
<action name="A" class="fancylab.hibiscidai.action.AAction">
</package>

访问方式:localhost:[port]/ProjectName/namespace/A

跨包间的跳转

1
2
3
4
<result name="D" type="chain">
<param name="namespace">/second</param><!--目标包的namespace-->
<param name="actionName">D</param><!--目标包下的目标Action-->
</result>

全局跳转

当许多Action跳转到相同路径时,可以定义全局跳转,减少配置文件当中的配置信息冗余。

1
2
3
4
5
<global-results>
<result name="success">
/ok.jsp
</result>
</global-results>

注意事项

  • 只在本包内有效
  • 局部配置优先

STRUTS2接收CLIENT的参数(重点)

收集客户端的零散数据

login.jsp

1
2
3
<input type="text" name="username">
<input type="password" name="password">
<input type="submit" value="login">
1
2
3
4
5
6
7
8
9
10
xxxAction implements Action {
private String username;
private String password;

//setget方法

public String execute() {
UserService.login(username, password);
}
}

好处

  • 简化了收集client数据的方式。
1
2
3
HttpServletRequest request = ServletActionContext.getRequest();
String username = request.getParameter("username
");
  • 对于通用数据进行自动类型转换。
  • 针对于post提交数据的方式,自动解决字符集编码问题。

通过对象收集客户端的数据

register.jsp

1
2
3
4
5
用户名<input type="text" name="user.username">
密码<input type="password" name="user.password">
年龄<input type="text" name="user.age">
日期<input type="date" name="user.birthdate">
<input type="submit">

User.java

1
2
3
4
5
6
7
8
class User {
private String username;
private String password;
private int age;
private Date birthdate;

//setter和getter方法
}

RegisterAction.java

1
2
3
4
5
6
7
RegisterAction implements Action {
private User user;
//setter和getter方法
public String execute() {
UserService.register(user);
}
}

通过数组或集合的形式收集客户端的数据

  • 批量删除
  • 用户勾选多个选项时
1
2
3
4
5
<input type="checkbox" value="1" name="a">
<input type="checkbox" value="2" name="a">
<input type="checkbox" value="3" name="a">

private List a;//以数组的形式接收数据

STRUTS2中ACTION的第二种开发方式

1
MyAction extends ActionSupport

DMI(DYNAMIC METHOD INVOKE 动态方法调用)(实战)

在一个Action中提供多个方法应对用户不同需求

编码

extends ActionSupport(建议)
语法要求:DMI中Action中的方法,方法名随便写
修饰符 返回值 参数列表 与execute中方法保持一致

配置

第一种配置

method 标签中设置方法
优点:可读性好
缺点:配置信息冗余

1
<action name="addUser" class="fancylab.hibiscidai.action.UserAction" method="add"></action>

第二种配置

采用通配符进行
优点:配置信息不再冗余
缺点:可读性极差

1
<action name="user_" class="fancylab.hibiscidai.action.UserAction" method="{1}"></action>

STRUTS2中的数据处理机制

数据处理机制:数据在网站中的流转
33648

OGNL表达式

OGNL表达式:是一种独立的表达式语言,不依赖于任何的框架
OGNL表达式特点: 从root区,ContextMap区取数据

从Root区取数据

从Root区中取所存对象的属性值

从Root去中取所存对象的属性值语法:直接属性名的方式

1
2
3
4
5
6
7
8
9
10
11
@Test
public void test1() throws Exception{
Person person = new Person();
person.setUsername("laowang");
person.setPassword("12345");
person.setAge(30);

System.out.println(Ognl.getValue("username",person));
System.out.println(Ognl.getValue("password",person));
System.out.println(Ognl.getValue("age",person));
}

从root区中取某一个对象中的关联对象的属性值:关联引用名.属性

1
2
3
4
5
6
7
8
9
10
11
12
13
14
@Test
public void test2() throws Exception{
Person person = new Person();
person.setUsername("laowang");
person.setPassword("12345");
person.setAge(30);
Address address = new Address();

address.setStreet("文化路");
person.setAddress(address);

System.out.println(Ognl.getValue("address.addressname",person));
System.out.println(Ognl.getValue("address.street",person));
}

从root区中取某一个对象当中的List集合中的元素:List集合引用名[下标]

1
2
3
4
5
6
7
8
9
10
11
12
13
14
@Test
public void test3() throws Exception{
Person person = new Person();
person.setUsername("laowang");
person.setPassword("12345");
person.setAge(30);

List<String> tels = person.getTels();
tels.add("xjr");
tels.add("whp");

System.out.println(Ognl.getValue("tels[0]",person));
System.out.println(Ognl.getValue("tels[1]",person));
}

从root区中取某一个对象当中的Map集合中的某一个元素:map集合的引用名[键]

1
2
3
4
5
6
7
8
9
10
11
12
@Test
public void test4() throws Exception{
Person person = new Person();
person.setUsername("laowang");
person.setPassword("12345");
person.setAge(30);
Map<String, String> qqs = person.getQqs();
qqs.put("kuaige","562471794");
qqs.put("zpf","7654321");
System.out.println(Ognl.getValue("qqs['kuaige']",person));
System.out.println(Ognl.getValue("qqs['zpf']",person));
}

ognl表达式中的运算

  • 算数运算 + - * / %
  • 比较运算 > < >= <= !=
  • 逻辑运算 && || !
1
2
3
4
5
6
7
8
9
10
@Test
public void test5() throws Exception{
Person person = new Person();
person.setUsername("laowang");
person.setPassword("12345");
person.setAge(30);

System.out.println(Ognl.getValue("age<10",person));
System.out.println(Ognl.getValue("username=='laowang'",person));
}

OGNL表达式可以调用某种数据类型的方法

1
2
3
4
5
6
7
8
9
@Test
public void test6() throws Exception{
Person person = new Person();
person.setUsername("laowang");
person.setPassword("12345");
person.setAge(30);
System.out.println(Ognl.getValue("username.toUpperCase()",person));
System.out.println(Ognl.getValue("username.equals('laowang')",person));
}

从ContextMap区取数据

contextmap本身是个map,在单独测试ognl时需要提供一个map集合
语法:#key的方式取值

1
2
3
4
5
6
7
8
9
10
11
//ContextMap区取值的方式
@Test
public void test1() throws Exception{
Map<String,Person> contextmap = new HashMap<String,Person>();
Person person = new Person();
person.setUsername("laowang");
person.setPassword("12345");
person.setAge(30);
contextmap.put("A", person);
System.out.println(Ognl.getValue("#A.age+10", contextmap,new Object()));
}

VALUESTACK

作用:管理(存储)一次请求有效的数据

  1. 客户端传来的数据
  2. 作用域中的数据
  • request
  • session
  • application

好处

与视图层(view层)解耦合

获取值栈

1
2
ActionContext ac = ActionContext.getContext();
ValueStack vs = ac.getValueStack();

值栈的生命周期(request作用域)

一次请求有效,请求变化则值栈变化

值栈的内存结构

请求格式

1
2
3
request.setAttribute("name", "laowang");
request.getAttribute("name");
request.setAttribute("n", "feige");

user类

1
2
3
4
class User {
private String username;
//setter和getter方法
}

request作用域底层

1
2
3
4
5
6
7
8
9
10
11
12
class request {
private Map attribute;

public void setAttribute(String name, String object) {
attribute.put(name, object);
}

public Object getAttribute(String name) {
Object = attrbute.get("name");
return object;
}
}

对于作用域
request—map
session—map
application—map

值栈的内存结构
50883

值栈的注意事项

问题:值栈是一次请求有效,为什么可以管理session application作用域?
51145

STRUTS中的标签(上)

作用:配合值栈在视图层显示数据
引用:
JSTL标签:

1
<%@ taglib prefix="c" uri="http://java.sun.com/jsp/jstl/core" %>

Strtus标签:

1
<%@taglib prefix="s" uri="/struts-tags"%>

显示数据

显示单个数据

1
<s:property value="OGNL表达式" />
  • 单一流程
1
2
3
4
<s:if test="OGNL表达式" />
</s:if>
<s:else>
</s:else>
  • 非单一流程
1
2
3
4
5
6
<s:if test="OGNL表达式">
</s:if>
<s:elseif test="OGNL表达式" />
</s:elseif>
<s:else>
</s:else>

显示多个数据

语法:

1
2
<s:iterator value="OGNL表达式">
</s:iterator>
从数组或集合中显示数据(对象类型)

List或Set数组

1
2
3
4
5
<s:iterator value="#request.users">
<h1><s:property value="username"/></h1>
<h1><s:property value="password"/></h1>
<h1><s:property value="age"/></h1>
</s:iterator>

Map

1
2
3
4
5
6
7
<s:iterator value="#request.users">
<!--取map的键-->
<h1><s:property value="key"/></h1>
<h1>==========</h1>
<!--取map的值-->
<h1><s:property value="value"/></h1>
</s:iterator>
从数组或集合中显示数据(String类型和8种基本类型)
1
2
3
<s:iterator value="#request.s">
<s:property/>
</s:iterator>
遍历状态
1
2
3
4
5
6
7
8
<s:iterator value="OGNL" status="s">
#s.count 遍历次数
#s.index 遍历的下标
#s.odd 是否是奇数遍历
#s.even 是否是偶次遍历
</s:iterator>
<s:iterator value="OGNL" begin="" end="" Step="">
</s:iterator>

begin:从某一个下标开始遍历
end:以某一个下标结束
step:步幅

查询所有的思路:
60321

STRUTS2数据处理机制的补充

Action中的成员变量替换request作用域

60576

Action中成员变量的作用:

  • 收集客户端的参数(零散变量,对象,数组或集合)
  • 替换request作用域

简化值栈操作session作用域,application作用域的开发

Struts2ScopeUtil工具类的开发

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
import com.opensymphony.xwork2.ActionContext;
import com.opensymphony.xwork2.util.ValueStack;

//简化值栈操作session作用域及application作用域的使用
public class Struts2ScopeUtil {

// 向session作用域中存值
public static void setSessionAttribute(String OGNL, Object value) {
ActionContext context = ActionContext.getContext();
ValueStack valueStack = context.getValueStack();
valueStack.setValue("#session." + OGNL, value);
}

// 从Session作用域中取值
public static Object getSessionAttribute(String OGNL) {
ActionContext context = ActionContext.getContext();
ValueStack valueStack = context.getValueStack();
return valueStack.findValue("#session." + OGNL);
}

// 向application作用域中存值
public static void setApplicationAttribute(String OGNL, Object value) {
ActionContext context = ActionContext.getContext();
ValueStack valueStack = context.getValueStack();
valueStack.setValue("#application." + OGNL, value);
}

// 从application作用域中取值
public static Object getApplicationAttribute(String OGNL) {
ActionContext context = ActionContext.getContext();
ValueStack valueStack = context.getValueStack();
return valueStack.findValue("#application." + OGNL);
}
}

为什么叫值栈?值栈体现在哪个区呢?

栈:先进后出

DEBUG使用

  • 打断点
  • tomcat以debug模式启动
  • F5进入方法内部/F6不进入方法内部,只显示程序流程/F8推出debug模式/类似于ArrayList的内存结构

66279

总结:现有STRUTS2的开发步骤

66507

STRUTS标签(续)

<s:date/>

语法:<s:date name="OGNL" format="自定义日期类型" />
作用:自定义日期的格式

1
<s:date name="#request.date" format="yyyy-MM-dd hh:mm:ss" />

<s:url/>

语法:<s:url action="" namespace=""/><s:url value=""/>
作用:防止用户禁用Cookie,自动进行url重写。加载第三方资源。
注意:传值<s:url action="" namespace=""/>?id=41

s标签中与html相关的UI标签

<s:form></s:form>—————><form></form>
<s:head/> —————><head></head>
<s:text name=""></s:text>—————><input type="text"/>
<s:date name=""/>—————><input type="date"/>

<s:action />

语法:<s:action name="" namespace="" executeResult="" />
作用:把多个Action的处理结果作整合

针对于前台视图

传统思路

67794

Struts解决方案

68013

STRUTS2中的拦截器

拦截器的作用:把多个ACTION中的冗余代码,抽取到拦截器中,解决代码冗余问题

使用

编码implements Interceptor接口

  • 方法作用:
    把多个Action中的冗余代码,写入次方法中,解决代码冗余问题
  • 参数的作用:
    ai.getAction();//获取目标的Action
    ai.getStack();//获取值栈
    ai.invoke();//控制请求的流程走向
  • 返回值的作用:
    中断用户请求时,指向跳转的目标JSP页面
1
2
3
4
5
6
7
8
9
10
11
12
13
public class checkloginInterceptor extends AbstractInterceptor {
@Override
public String intercept(ActionInvocation ai) throws Exception {
// 获取登陆或注册的标志位
String flag = (String) Struts2ScopeUtil.getSessionAttribute("flag");
if (flag == null) {
return "login";
} else {
ai.invoke();
}
return null;
}
}

配置

1
2
3
4
5
6
<interceptors>
<interceptor name="myInterceptor" class="fancylab.hibiscidai.interceptor.MyInterceptor">
</interceptors>
<action name="A" class="fancylab.hibiscidai.action.AAction" method="A">
<interceptor-ref name=myinterceptor"></interceptor-ref>
</action>

注意:

拦截响应

72101

中断请求

72327

拦截器只在本包中有效

简化拦截器开发 继承AbstractInterceptor类

拦截器栈

作用:管理多个拦截器
使用:不编码,只需配置

1
2
3
4
5
<!--拦截器栈-->
<interceptor-stack name="my">
<interceptor-ref name="myInterceptor"></interceptor-ref>
<interceptor-ref name="myInterceptor2"></interceptor-ref>
</interceptor-stack>

默认拦截器栈

作用:可以指定一个拦截器栈为默认拦截器栈,可以拦截所有的目标Action

1
<default-interceptor-ref name="my"></default-interceptor-ref>

注意:默认拦截器栈放置的位置必须在全局跳转的前边,每个包中只能配置一个默认拦截器,局部配置优先。

Struts2中的拦截器体系

自定义拦截器

默认拦截器(系统拦截器)了解

  • params

目的:接收客户端的请求参数

74527

  • fileupload
  • Exception
  • workflow

默认拦截器放置的位置

Struts2-core.jar ——>Struts-default.xml

74832

注意:如果自定义了默认拦截器栈,Struts2中的系统拦截器栈将失效

1
2
3
4
5
6
7
<interceptors>
<interceptor-stack name="my">
<interceptor-ref name="defalutStack"></interceptor-ref><!--引入系统拦截器-->
<interceptor-ref name="myInterceptor"></interceptor-ref>
<interceptor-ref name="myInterceptor2"></interceptor-ref>
</interceptor-stack>
</interceptors>

注意:如果自定义拦截器,系统拦截器将失效

方法拦截器

  • 作用
    在DMI中,如果使用者采用的是通配符的配置方式,可以通过方法拦截器,拦截对应的方法。
  • 编码
    extends MethodFilterInterceptor
1
2
3
4
5
6
7
8
public class MethodInterceptor extends MethodFilterInterceptor  {
@Override
public String doIntercept(ActionInvocation ai) throws Exception {
System.out.println("我是方法拦截器");
ai.invoke();
return null;
}
}
  • 配置
    拦截排除某些方法
1
2
3
<param name="includeMethod">
拦截器拦截哪些方法
</param>
1
2
3
4
5
6
7
8
9
10
11
12
13
<interceptors>
<interceptor name="methodinterceptor" class="fancylab.hibiscidai.interceptor.MethodInterceptor">
<param name="excludeMethods">
m4
</param>
</interceptor>
<interceptor-stack name="my">
<interceptor-ref name="defaultStack"></interceptor-ref>
<interceptor-ref name="methodinterceptor"></interceptor-ref>
</interceptor-stack>
</interceptors>
<defalut-interceptor-ref name="my">
</defalut-interceptor-ref>
  • 注意
    81538
  • 总结
    拦截器的开发步骤
    81757

拦截器的应用

强制登陆

81986

防止用户重复提交(令牌环)

  • 发生场景
    在用户进行表单提交时,因为网络通信等问题,产生重复的表单提交
  • 解决方案
    令牌环
  • 令牌环实现原理
    82280

Struts2令牌环解决思路

客户端生成随机数

1
<s:token></s:token>

配置token拦截器

引入Struts2提供的token拦截器

1
2
3
4
<interceptor-stack name="my1">
<interceptor-ref name="token"></interceptor-ref>
<interceptor-ref name="defaultStack"></interceptor-ref>
</interceptor-stack>

提供跳转目标页面

1
2
3
<result name="invalid.token">
/error.jsp
</result>

STRUTS2中的高级部分

上传

文件上传核心思路

  • client问题
  • 服务器端如何获得文件上传的内容
  • 服务器端如何存储文件上传的内容

84748

文件上传开发步骤

客户端的处理

enctype="application/x-www-form-urlencoded" 把表单中的文本中的内容,提交到服务器中

1
2
<form method="post" action="" enctype="application/x-www-form-urlencoded">
</form>

enctype="multipart/form-data" 告知服务器端识别客户端传入的文件内容

1
2
3
4
<form method="post" action="" enctype="multipart/form-data">
<input type="file" name="upload" />
<input type="submit" value="上传" />
</form>

服务器端创建一个文件夹,用于保存用户上传的文件

处理客户端上传的文件

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
public class uploadAction extends ActionSupport {
private File upload;
// 获取客户端传入的文件名和后缀
private String uploadFileName;
// 获取客户端传入的文件类型
private String uploadContentType;

// 文件存放的目录
private String directory;

public String getDirectory() {
return directory;
}

public void setDirectory(String directory) {
this.directory = directory;
}

public String getUploadContentType() {
return uploadContentType;
}

public void setUploadContentType(String uploadContentType) {
this.uploadContentType = uploadContentType;
}

public String getUploadFileName() {
return uploadFileName;
}

public void setUploadFileName(String uploadFileName) {
this.uploadFileName = uploadFileName;
}

public File getUpload() {
return upload;
}

public void setUpload(File upload) {
this.upload = upload;
}

public String upload() {
/*InputStream is = null;
OutputStream os = null;
// 把文件读入IO流
try {
is = new FileInputStream(upload);
// 把文件写出到文件系统中
os = new FileOutputStream("D:\\Class3\\apache-tomcat-7.0.67\\webapps\\Strutspratice\\upload\\zkf.txt");
byte[] buffer = new byte[1024];
int len = 0;
while (true) {
// 读操作
len = is.read(buffer, 0, buffer.length);
if (len == -1)
break;
// 写操作
os.write(buffer, 0, len);
}
return "uploadOK";
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
return "uploadError";
} finally {
try {
// 关闭流
is.close();
os.close();
} catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}*/
try {
System.out.println(uploadContentType);
System.out.println(directory);
FileUtils.copyFile(upload, new File(getRealPath(directory) + "\\" + uploadFileName));
return "uploadOK";
} catch (Exception e) {
// TODO Auto-generated catch block
e.printStackTrace();
return "uploadError";
}
}

public String getRealPath(String path) {
ServletContext servletContext = ServletActionContext.getServletContext();
// 获取绝对路径
String realPath = servletContext.getRealPath(path);
return realPath;

}
}

文件上传(重构)

IO操作过于频繁,希望简化IO的处理

Commons-io.jar 提供文件操作的工具类

1
FileUtils.copyFile(upload, new File(getRealPath(directory) + "\\" + uploadFileName));
文件名

如何获得用户上传的文件名

1
2
3
4
5
private File upload;
// 获取客户端传入的文件名和后缀
private String uploadFileName;
// 获取客户端传入的文件类型
private String uploadContentType;
文件路径

如何在web开发中通过相对路径获取绝对路径

1
2
3
ServletContext sc = ServletActionContext.getServletContext();
String RealPath = ac.getRealPath(“相对路径”);
String RealPath = ac.getRealPath(“/upload”);
维护性差

文件目录转移到配置文件中进行配置

1
2
3
<action name="upload" class="fancylab.hibiscidai.Action.uploadAction" method="upload">
<param name="directory>/upload</param>
</action>

Action中声明成员变量即可

1
private String directory;
Struts2中上传的文件默认大小为2M

101838

1
2
<!--指定Struts2中 文件上传大小的上限-->
<constant name="struts.multipart.maxSize" value="2097152000"></constant>

下载

/struts-2.3.15.1/docs/WW/docs/stream-result.html stream-result

文件下载的核心思路

102842

Struts2下载的步骤

extends ActionSupport

1
2
3
4
//目的告知Struts2将要下载的文件
public InputStream getInputStream() throws Exception {
return new FileInputStream(getRealPath(directory) + "\\" + filename);
}
1
2
3
4
5
6
<result type="stream">
<!-- 指定文件下载类型 -->
<param name="contentType">text/plain</param>
<!-- inline代表浏览器下载之后直接打开 attachment以附件的形式进行下载 filename下载之后的文件名 -->
<param name="contentDisposition">attachment;filename=${filename}</param>
</result>

文件下载重构(优化)

完成路径的修改

通过getRealPath()方法与Action中为成员变量赋值的方式,完成路径的修改。

下载中如何处理用户需要下载的文件名字

客户通过传递参数的形式,向Action中传递数据,Action中通过成员变量接收数据。

1
2
3
4
5
6
7
8
9
10
11
12
<ul>
<li>
<a href="<s:url action='download' namespace='/user' />?filename=z.txt">
z.txt
</a>
</li>
<li>
<a href="<s:url action='download' namespace='/user' />?filename=zkf.txt">
zkf.txt
</a>
</li>
</ul>
1
2
//获取客户端所要下载的文件名
private String filename;
解决用户下载之后的文件名
1
2
//获取客户端所要下载的文件名
private String filename;
1
2
3
4
5
6
7
8
9
<action name="download" class="com.baizhi.Action.downloadAction">
<param name="directory">/upload</param>
<result type="stream">
<!-- 指定文件下载类型 -->
<param name="contentType">text/plain</param>
<!-- inline代表浏览器下载之后直接打开 attachment以附件的形式进行下载 filename下载之后的文件名 -->
<param name="contentDisposition"> attachment;filename=${filename}</param>
</result>
</action>
文章作者: HibisciDai
文章链接: http://hibiscidai.com/2018/03/30/2018-03-30-Struts2学习笔记/
版权声明: 本博客所有文章除特别声明外,均采用 CC BY-NC-SA 4.0 许可协议。转载请注明来自 HibisciDai
好用、实惠、稳定的梯子,点击这里