分享
 
 
 

Servlet 2.3过滤器编程(二)

王朝java/jsp·作者佚名  2006-01-08
窄屏简体版  字體: |||超大  

Servlet 2.3过滤器编程(二)

文件上传过滤器

我们将看到的最后一个过滤器是处理多路/多类型数据的POST请求,该类型的请求能包含文件上传。每个多路/多类型数据POST请求包含所有参数和文件,使用一种servlet不能识别的特别的格式。历史上Servlet开发者使用第三方类来处理上传,例如在我的com.oreilly.servlet包中的MultipartRequest和MultipartParser类。这里我们将看到一种使用MultipartFilter的新方法来是处理这种请求更容易。该过滤器基于com.oreilly.servlet包下的parsers构建并已经被集成到该包中(参见Resources)。

MultipartFilter工作与观察输入的请求,当它发现一个文件上传请求时(content type:multipart/form-data),过滤器使用一个知道如何分析这种特殊content type格式的特殊请求包来将请求对象进行包装。servlet获取此特殊请求包并通过标准的getParameter()方法来无缝地访问此multipart参数,因为这个wrapper中已经重新定义了这些方法的功能。此servelt能够通过将requset转换成wrapper类型并使用wrapper中附加的getFile()方法来处理文件上传。

过滤器代码:

package com.oreilly.servlet;

import java.io.*;

import javax.servlet.*;

import javax.servlet.http.*;

public class MultipartFilter implements Filter {

private FilterConfig config = null;

private String dir = null;

public void init(FilterConfig config) throws ServletException {

this.config = config;

// Determine the upload directory. First look for an uploadDir filter

// init parameter. Then look for the context tempdir.

dir = config.getInitParameter("uploadDir");

if (dir == null) {

File tempdir = (File) config.getServletContext()

.getAttribute("javax.servlet.context.tempdir");

if (tempdir != null) {

dir = tempdir.toString();

}

else {

throw new ServletException(

"MultipartFilter: No upload directory found: set an uploadDir " +

"init parameter or ensure the javax.servlet.context.tempdir " +

"directory is valid");

}

}

}

public void destroy() {

config = null;

}

public void doFilter(ServletRequest request, ServletResponse response,

FilterChain chain) throws IOException, ServletException {

HttpServletRequest req = (HttpServletRequest) request;

String type = req.getHeader("Content-Type");

// If this is not a multipart/form-data request continue

if (type == null || !type.startsWith("multipart/form-data")) {

chain.doFilter(request, response);

}

else {

MultipartWrapper multi = new MultipartWrapper(req, dir);

chain.doFilter(multi, response);

}

}

}

init()方法确定文件上传的路径。这是multipart parser放置文件的地方,因此实际的请求并不需要驻留在内存中。它先查找uploadDir过滤器初始化参数,如果没找到,则使用默认的tempdir目录——Servlet API 2.2中的标准context属性。

doFilter()方法检查请求的content type,如果是multipart/form-data请求,则将此请求用MultipartWrapper打包。wrapper代码如下:

package com.oreilly.servlet;

import java.io.*;

import java.util.*;

import javax.servlet.*;

import javax.servlet.http.*;

public class MultipartWrapper extends HttpServletRequestWrapper {

MultipartRequest mreq = null;

public MultipartWrapper(HttpServletRequest req, String dir)

throws IOException {

super(req);

mreq = new MultipartRequest(req, dir);

}

// Methods to replace HSR methods

public Enumeration getParameterNames() {

return mreq.getParameterNames();

}

public String getParameter(String name) {

return mreq.getParameter(name);

}

public String[] getParameterValues(String name) {

return mreq.getParameterValues(name);

}

public Map getParameterMap() {

Map map = new HashMap();

Enumeration enum = getParameterNames();

while (enum.hasMoreElements()) {

String name = (String) enum.nextElement();

map.put(name, mreq.getParameterValues(name));

}

return map;

}

// Methods only in MultipartRequest

public Enumeration getFileNames() {

return mreq.getFileNames();

}

public String getFilesystemName(String name) {

return mreq.getFilesystemName(name);

}

public String getContentType(String name) {

return mreq.getContentType(name);

}

public File getFile(String name) {

return mreq.getFile(name);

}

}

wrapper构造了一个com.oreilly.servlet.MultipartRequest对象以处理上传分析并重载getParameter()方法族以使用MultipartRequest取代未加工的请求来读取参数值。wrapper也定义了不同的getFile()方法以使一个servlet接收打包过的请求以能通过调用其他方法来处理上传的文件。

Web.xml部署描述器用如下代码来添加此过滤器:

<filter>

<filter-name>multipartFilter</filter-name>

<filter-class>com.oreilly.servlet.MultipartFilter</filter-class>

<!--

<init-param>

<param-name>uploadDir</param-name>

<param-value>/tmp</param-value>

</init-param>

-->

</filter>

<filter-mapping>

<filter-name>multipartFilter</filter-name>

<url-pattern>/*</url-pattern>

</filter-mapping>

<servlet>

<servlet-name>

uploadTest

</servlet-name>

<servlet-class>

UploadTest

</servlet-class>

</servlet>

<servlet-mapping>

<servlet-name>

uploadTest

</servlet-name>

<url-pattern>

/uploadTest

</url-pattern>

</servlet-mapping>

UploadText servlet如下:

import java.io.*;

import java.util.*;

import javax.servlet.*;

import javax.servlet.http.*;

import com.oreilly.servlet.*;

public class UploadTest extends HttpServlet {

public void doPost(HttpServletRequest req, HttpServletResponse res)

throws ServletException, IOException {

res.setContentType("text/html");

PrintWriter out = res.getWriter();

out.println("<HTML>");

out.println("<HEAD><TITLE>UploadTest</TITLE></HEAD>");

out.println("<BODY>");

out.println("<H1>UploadTest</H1>");

// Parameters can now be read the same way for both

// application/x-www-form-urlencoded and multipart/form-data requests!

out.println("<H3>Request Parameters:</H3><PRE>");

Enumeration enum = req.getParameterNames();

while (enum.hasMoreElements()) {

String name = (String) enum.nextElement();

String values[] = req.getParameterValues(name);

if (values != null) {

for (int i = 0; i < values.length; i++) {

out.println(name + " (" + i + "): " + values[i]);

}

}

}

out.println("</PRE>");

// Files can be read if the request class is MultipartWrapper

// Init params to MultipartWrapper control the upload handling

if (req instanceof MultipartWrapper) {

try {

// Cast the request to a MultipartWrapper

MultipartWrapper multi = (MultipartWrapper) req;

// Show which files we received

out.println("<H3>Files:</H3>");

out.println("<PRE>");

Enumeration files = multi.getFileNames();

while (files.hasMoreElements()) {

String name = (String)files.nextElement();

String filename = multi.getFilesystemName(name);

String type = multi.getContentType(name);

File f = multi.getFile(name);

out.println("name: " + name);

out.println("filename: " + filename);

out.println("type: " + type);

if (f != null) {

out.println("length: " + f.length());

}

out.println();

}

out.println("</PRE>");

}

catch (Exception e) {

out.println("<PRE>");

e.printStackTrace(out);

out.println("</PRE>");

}

}

out.println("</BODY></HTML>");

}

}

servlet的前一半显示了过滤器如何将参数数据毫无改变的传送给接收的servlet。后半部分则显示了一个servlet是如何将请求向下传送给MultipartWrapper以获得附加的文件访问方法。

一个驱动此servlet的HTML例子如下:

<FORM ACTION="uploadTest" ENCTYPE="multipart/form-data" METHOD=POST>

What is your name? <INPUT TYPE=TEXT NAME=submitter> <BR>

What is your age? <INPUT TYPE=TEXT NAME=age> <BR>

Which file do you want to upload? <INPUT TYPE=FILE NAME=file1> <BR>

Any other file to upload? <INPUT TYPE=FILE NAME=file2> <BR>

<INPUT TYPE=SUBMIT>

</FORM>

这是可能的输出:

UploadTest

Request Parameters:

submitter (0): Jason

age (0): 28

Files:

name: file1

filename: 4008b21.tif

type: application/octet-stream

length: 39396

name: file2

filename: null

type: null

你们也许会疑惑我们如何确信由过滤器设置的MultipartWrapper能正确的传送给接下来的servlet。在2号发布稿规范和Tomcat 4.0 beta 5中,那是不确定的。实际上,如果你试图用/servlet/UploadTest去访问此servlet,你将注意到过滤并没有正确工作,因为/servlet的调用将MultipartWrapper转给Tomcat自己的特殊wrapper了。这允许参数能被正确解析,但文件访问方法却不能正确工作。在Servlet API专家组的讨论中,我们决定servlet容器不能对过滤器的wrapper做更多的包装。这个servlet规范将被修改的更清楚。在Tomcat 4.0以后版本中将澄清这些规则。短期的办法是在请求wrapper中使用getRequest()方法来“沿着”请求去找到这个隐藏的multipart wrapper。

从如下地址下载该WAR文件:

http://www.javaworld.com/jw-06-2001/Filters/mulitpart.war

过滤器能力

Servlet过滤器提供了一中强大的能力来控制请求的操作和响应的发生,提供新的servlet功能而不需要太多的代码。我希望通过这些已经向你展示使用过滤器的可能情况,并教给你一些关于如何更有效的使用新的过滤器机能的技巧。

感谢这些过滤器的作者以及其他对过滤器提供有用建议的的人:Amy Roh,Criag McClanahan,Serge Knystautas,以及OpenSymphony成员。

关于作者

见原文

资源

见原文

file://请达人给指点斧正了,谢谢:)

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
 
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有