深入分析Java中webwork的文件上传机制

2016-02-19 20:03 20 1 收藏

下面这个深入分析Java中webwork的文件上传机制教程由图老师小编精心推荐选出,过程简单易学超容易上手,喜欢就要赶紧get起来哦!

【 tulaoshi.com - 编程语言 】

  点击上传按钮后,webwork的程序流如下:

  step 1)进入ServletDispatcher.service

public void service(HttpServletRequest request, HttpServletResponse response)
  throws ServletException {
  ........
  request = wrapRequest(request);
  .........
  }

(本文来源于图老师网站,更多请访问https://www.tulaoshi.com/bianchengyuyan/)

  step2)进入ServletDispatcher.wrapRequest

protected HttpServletRequest wrapRequest(HttpServletRequest request)
  throws IOException {
  ........................
  if (MultiPartRequest.isMultiPart(request)) {
  request = new MultiPartRequestWrapper(request, getSaveDir(), getMaxSize());
  }
  return request;
  }

  step3)进入MultiPartRequestWrapper的构造方法

public MultiPartRequestWrapper(HttpServletRequest request, String saveDir, int maxSize)
  throws IOException {
  .....................
  //step3.1)获取webwork.preperties配置的parser
  String parser = "";
  parser = Configuration.getString("webwork.multipart.parser");
  // If it's not set, use Pell
  if (parser.equals("")) {
  log.warn("Property webwork.multipart.parser not set." +
  " Using com.opensymphony.webwork.dispatcher.multipart.PellMultiPartRequest");
  parser = "com.opensymphony.webwork.dispatcher.multipart.PellMultiPartRequest";
  }
  // legacy support for old style property values
  else if (parser.equals("pell")) {
  parser = "com.opensymphony.webwork.dispatcher.multipart.PellMultiPartRequest";
  } else if (parser.equals("cos")) {
  parser = "com.opensymphony.webwork.dispatcher.multipart.CosMultiPartRequest";
  } else if (parser.equals("jakarta")) {
  parser = "com.opensymphony.webwork.dispatcher.multipart.JakartaMultiPartRequest";
  }
  //step3.2)获取后通过反射实例化parser
  try {
  Class baseClazz =
  com.opensymphony.webwork.dispatcher.multipart.MultiPartRequest.class;
  Class clazz = Class.forName(parser);
  // make sure it extends MultiPartRequest
  if (!baseClazz.isAssignableFrom(clazz)) {
  addError("Class '" + parser + "' does not extend MultiPartRequest");
  return;
  }
  // get the constrUCtor
  Constructor ctor = clazz.getDeclaredConstructor(new Class[]{
  Class.forName("Javax.servlet.http.HttpServletRequest"),
  java.lang.String.class, int.class
  });
  // build the parameter list
  Object[] parms = new Object[]{
  request, saveDir, new Integer(maxSize)
  };
  // instantiate it
  multi = (MultiPartRequest) ctor.newInstance(parms);
  .................................................
  }

  step4 进入JakartaMultiPartRequest的构造方法(在webwork配置的parser是Jakarta所以进入了这个方法,假如你配置不同的parser回进入不同的parser

public JakartaMultiPartRequest(HttpServletRequest servletRequest, String saveDir, int maxSize)
  throws IOException {
  //设置保存参数
  DiskFileUpload upload = new DiskFileUpload();
  // we must store all uploads on disk because the ww multipart API is missing streaming
  
   // capabilities
  upload.setSizeThreshold(0);
  upload.setSizeMax(maxSize);
  if (saveDir != null) {
  upload.setRepositoryPath(saveDir);
  }
  // Parse the request
  try {
  /**此方法生成文件,将请求中的每个参数都生成一个
  *临时文件比如upload_00000017.tmp, upload_00000018.tmp等,
  *就算是form提交的参数也如此
  */
  List items = upload.parseRequest(servletRequest);
  ......................
  }

  执行完第四步,然后推出ServletDispatcher.wrapRequest,进入serviceAction方法,开始action及其拦截器的栈调用

  在此过程中会删除非上传文件的临时文件,至于哪一步删除,我还没看出来,有时候很早有时候很晚,有时候甚至没有删除,我怀疑有个dameon在做这个事。

  进入action和调用栈后,拦截器或action可通过如下代码访问上传的临时文件MultiPartRequestWrapper wrapper = (MultiPartRequestWrapper) req;

(本文来源于图老师网站,更多请访问https://www.tulaoshi.com/bianchengyuyan/)

File doc = wrapper.getFiles("doc")[0];

  从上面的分析可以看出:

  1)假如你使用webwork来上传文件(在进入action栈之前不修改源码或者做一些扩展、覆盖之类的动作),在进入action栈的时候文件已经上 传,而且其文件名很难跟踪(upload_00000017.tmp,到底是00000017,0000018,或者0000022等等),究竟有很多人 上传文件,所以临时文件名很难确定,所以假如你想知道上传的进度很难。

  2)利用webwork上传文件是两次拷贝过程,webwork首先从request的输入流中将文件流输出到一个临时文件,然后你再将此临时文件拷贝到你需要指定的路径。


来源:https://www.tulaoshi.com/n/20160219/1622640.html

延伸阅读
首先,当考虑到内存分配和执行性能的时候,使用std::deque要比std::vector好。 Deque总览 deque和vector一样都是标准模板库中的内容,deque是双端队列,在接口上和vector非常相似,在许多操作的地方可以直接替换。假如读者已经能够有效地使用vector容器,下面提供deque的成员函数和操作,进行对比参考。 函数描述c.assign(beg,end) c.assign(...
首先看一个例子: 代码如下: #include iostream  using namespace std;  class A{};  class B  {      int b;      char c;  };  class C  {      int c1;         static int c2;  }; &nbs...
Struts和JSF/Tapestry都属于表现层框架,这两种分属不同性质的框架,后者是一种事件驱动型的组件模型,而Struts只是单纯的MVC模式框架,老外总是急吼吼说事件驱动型就比MVC模式框架好,何以见得,我们下面进行详细分析比较一下到底是怎么回事? 首先事件是指从客户端页面(浏览器)由用户操作触发的事件,Struts使用Action来接受浏览器表单提...
java的比较器有两类,分别是Comparable接口和Comparator接口。 在为对象数组进行排序时,比较器的作用非常明显,首先来讲解Comparable接口。 让需要进行排序的对象实现Comparable接口,重写其中的compareTo(T o)方法,在其中定义排序规则,那么就可以直接调用java.util.Arrays.sort()来排序对象数组,实例如下: 代码如下: class Student im...
通常低级事件的处理是为了满足游戏开发的需要,因为游戏开发相关的问题非常复杂。因此不在这里进行阐述,有爱好的朋友可以和我索要相关的游戏开发资料。MIDP定义了如下键值在Canvas类内: KEY_NUM0 KEY_NUM1 KEY_NUM2 KEY_NUM3 KEY_NUM4 KEY_NUM5 KEY_NUM6 KEY_NUM7 KEY_NUM8 KEY_NUM9 KEY_ST...

经验教程

416

收藏

61
微博分享 QQ分享 QQ空间 手机页面 收藏网站 回到头部