Struts2多个文件上传的分析及源代码说明


多个文件上传分为List集合和数组,下面我们着重介绍一下list集合的上传。都大同小异。



一  介绍

1.  在struts2文件上传的时候要先导入struts2的核心的几个包,包括
commons-fileupload.jar
commons-io.jar


2. Struts2文件上传并未提供自己的请求解析器,也就是说,struts2不会自己去处理multipart/form-data的请求,它需要调用其他的请求解析器,将http请求中的表单域解析出来。但struts2在原有的上传解析器继承上做了进一步封装,更进一步简化了文件上传。


3.  Struts2默认使用的是Jakarta和Connon-FileUpload的文件上传框架,因此,如果需要使用struts2的文件上传功能,则需要在Web应用导入上面我所说的几个包


4.  Struts2的文件上传支持在原有的问上传项目上做了进一步封装,简化了文件上传代码实现,取消了不同上传项目上编程差异。


二  实例


1. 首先我们来了解一下表单属性enctype属性的意义


表单的enctype属性指定的是表单数据的编码方式,该属性呢有3个值


(1)       application/x-www-form-urlencoded,这是默认的编码方式,它只能处理表单域里的value属性,采用这种编码方式的表单会将表单域的值处理成URL编码方式。


(2)       multipart/form-data,采用这种编码方式会以二进制流的方式来处理表单数据 ,这种编码方式会把文件域指定文件的内容也封装到请求参数里。


(3)       text/plain,这种编码方式当表单的action属性为mailto:URL的形式是比较方便,这种方式主要适用于直接通过表单发送邮件的方式。


从以上的介绍可以看出为什么文件上传要用到的是multipart/form-data属性了吧!上传的文件会在底层封装,并通过二进制流读取。


2. 下面我们来写这样一个界面来实现多文件的上传页面:

所用的html代码为:

<%@ page language="java" import="java.util.*" pageEncoding="utf-8"%>  
  
<%@ taglib uri="/struts-tags" prefix="s"%>  
  
<%  
    String path = request.getContextPath();  
  
    String basePath = request.getScheme() + "://"  
  
           + request.getServerName() + ":" + request.getServerPort()  
  
           + path + "/";  
%>  
  
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">  
  
<html>  
  
    <head>  
  
       <base href="<%=basePath%>">  
  
       <title>My JSP 'tagUpload.jsp' starting page</title>  
  
       <meta http-equiv="pragma" content="no-cache">  
  
       <meta http-equiv="cache-control" content="no-cache">  
  
       <meta http-equiv="expires" content="0">  
  
       <meta http-equiv="keywords" content="keyword1,keyword2,keyword3">  
  
       <meta http-equiv="description" content="This is my page">  
  
       <!-- 
 
    <link rel="stylesheet" type="text/css" href="styles.css"> 
 
    -->  
    </head>  
    <body>  
      <h3>多个文件上传实例</h3>  
       <s:form action="/csdn/uploadList.action" enctype="multipart/form-data" method="post">  
           <s:textfield label="上传名称" name="name"></s:textfield>  
           <s:file label="上传文件" name="upload"></s:file>  
           <s:file label="上传文件" name="upload"></s:file>  
           <s:file label="上传文件" name="upload"></s:file>  
           <s:submit value="上传" />  
       </s:form>  
    </body>  
</html>   

上面的页面只是一个普通的html页面,没有任何的动态部分,当该页面提交请求的时候发送到/csdn/uploadList.action,这是一个struts2的action。


    Struts2的Action无需负责处理HttpServletRequest请求,因为struts2的Action已经与servletAPI彻底分离了,struts2框架负责解析httpServletRequest请求的参数,包括文件域,strtus2使用File类型来封装文件域。


3.下面是处理Action的代码:

package cn.csdn.hr.up.action;  
import java.io.File;  
import java.io.IOException;  
import java.util.List;  
import org.apache.commons.io.FileUtils;  
import org.apache.struts2.ServletActionContext;  
import com.opensymphony.xwork2.ActionSupport;  
public class TagUploadListAction extends ActionSupport {  
    private static final long serialVersionUID = 1L;  
    private String name;  
  
    // 上传多个文件的集合文本  
  
    private List<File> upload;  
    // /多个上传文件的类型集合  
    private List<String> uploadContextType;  
   // 多个上传文件的文件名集合  
    private List<String> uploadFileName;  
  
    public String getName() {  
            return name;  
     }  
  
    public void setName(String name) {  
  
       this.name = name;  
    }  
  
    public List<File> getUpload() {  
  
       return upload;  
    }  
  
    public void setUpload(List<File> upload) {  
  
       this.upload = upload;  
    }  
  
    public List<String> getUploadContextType() {  
  
       return uploadContextType;  
    }  
  
    public void setUploadContextType(List<String> uploadContextType) {  
  
       this.uploadContextType = uploadContextType;  
    }  
  
    public List<String> getUploadFileName() {  
  
       return uploadFileName;  
    }  
  
    public void setUploadFileName(List<String> uploadFileName) {  
  
       this.uploadFileName = uploadFileName;  
    }  
  
    public String execute() {  
  
       // 把上传的文件放到指定的路径下  
  
       String path = ServletActionContext.getServletContext().getRealPath(  
  
              "/WEB-INF/uploadList");  
  
       // 写到指定的路径中  
  
       File file = new File(path);  
  
       // 如果指定的路径没有就创建  
  
       if (!file.exists()) {  
  
           file.mkdirs();  
       }  
  
       // 把得到的文件的集合通过循环的方式读取并放在指定的路径下  
  
       for (int i = 0; i < upload.size(); i++) {  
           try {  
  
              //list集合通过get(i)的方式来获取索引  
  
              FileUtils.copyFile(upload.get(i), new File(file, uploadFileName.get(i)));  
  
           } catch (IOException e) {  
  
              // TODO Auto-generated catch block  
  
              e.printStackTrace();  
           }  
       }  
  
       return SUCCESS;  
    }  
}  



通过以上的Action我们可以看出Action还包括了两个属性,uploadFileName,uploadContextType, 这两个属性分别用来封装上传我文件的文件名,上传文件的文件类型,这两个属性体现了struts设计的灵巧、简化之处,Action类直接通过File类型属性直接封装了上传文件的内容,但这个File属性无法获取上传文件的文件名和文件类型,所以struts2直接将包含的上传文件名和文件类型的信息封装到uploadFileName,uploadContextType属性中,可以认为:如果表单中包含一个name属性为xxx的文件域,则对应的Action需要使用3个属性来封装该文件域的信息:


(1)   类型为File的xxx属性封装了该文件域对应的文件内容


(2)   类型为String的xxxFileName属性封装了该案文件域对应的文件的文件类型


(3)   类型为String的xxxContextType属性封装了该文件域对应的文件的类型


通过上吗的三个属性可以简单的实现上传文件的文件名、文件类型和文件内容


 
3. 配置action的strtus.xml

<!DOCTYPE struts PUBLIC  
  
    "-//Apache Software Foundation//DTD Struts Configuration 2.3//EN"  
  
    "http://struts.apache.org/dtds/struts-2.3.dtd">  
  
<struts>  
    <package name="file" extends="struts-default" namespace="/csdn">  
  
       <action name="uploadList" class="cn.csdn.hr.up.action.TagUploadListAction">  
  
           <result>../success.jsp</result>  
  
           <result name="input">../tagUpload.jsp</result>  
  
       </action>  
    </package>  
</struts>  

4. 设置上传的文件的大小和类型


就是在struts.xml中用拦截器来设置   


[html] view plaincopy
<action name="uploadList" class="cn.csdn.hr.up.action.TagUploadListAction">  
  
           <result>../success.jsp</result>  
  
           <result name="input">../tagUpload.jsp</result>  
  
           <!-- 通过拦截器来限制上传图片的类型和大小 -->  
  
           <interceptor-ref name="fileUpload">  
  
              <param name="allowedTypes">image/bmp,image/x-png,image/gif</param>  
  
              <param name="maximumSize">200</param>  
  
           </interceptor-ref>  
  
           <interceptor-ref name="defaultStack"></interceptor-ref>  
  
       </action>  
  
   
显示的错误提示信息的标签为:


             <s:fielderror></s:fielderror>


以上的多个文件上传是List集合的,数组的也不过如此,要改变的地方为Action接收的时候类型的不同和读取的时候循环不同,下面的数组的例子为:

// 得到上传文件的名称一定与name值一直  
private File upload[];  
// 上传文件的类型 ContentType  
private String uploadContentType[];  
// 上传文件的名称  
private String uploadFileName[];  
public File[] getUpload() {  
    return upload;  
}  
  
public void setUpload(File[] upload) {  
    this.upload = upload;  
}  
  
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 static long getSerialversionuid() {  
    return serialVersionUID;  
}  
  
public String uploads() {  
    String path = ServletActionContext.getServletContext().getRealPath(  
            "/upload");  
  
    // 写到指定路径  
    File file = new File(path);  
    //判断指定的路径下是否有uplaod,如果没有,自动创建  
    if (!file.exists()) {  
        file.mkdirs();  
    }  
    try {  
        for(int i = 0;i<upload.length;i++){  
            FileUtils.copyFile(upload[i], new File(file, uploadFileName[i]));  
        }  
    } catch (IOException e) {  
        // TODO Auto-generated catch block  
        e.printStackTrace();  
    }  



深度学习是机器学习的一个子领域,它基于人工神经网络的研究,特别是利用多层次的神经网络来进行学习和模式识别。深度学习模型能够学习数据的高层次特征,这些特征对于图像和语音识别、自然语言处理、医学图像分析等应用至关重要。以下是深度学习的一些关键概念和组成部分: 1. **神经网络(Neural Networks)**:深度学习的基础是人工神经网络,它是由多个层组成的网络结构,包括输入层、隐藏层和输出层。每个层由多个神经元组成,神经元之间通过权重连接。 2. **前馈神经网络(Feedforward Neural Networks)**:这是最常见的神经网络类型,信息从输入层流向隐藏层,最终到达输出层。 3. **卷积神经网络(Convolutional Neural Networks, CNNs)**:这种网络特别适合处理具有网格结构的数据,如图像。它们使用卷积层来提取图像的特征。 4. **循环神经网络(Recurrent Neural Networks, RNNs)**:这种网络能够处理序列数据,如时间序列或自然语言,因为它们具有记忆功能,能够捕捉数据中的时间依赖性。 5. **长短期记忆网络(Long Short-Term Memory, LSTM)**:LSTM 是一种特殊的 RNN,它能够学习长期依赖关系,非常适合复杂的序列预测任务。 6. **生成对抗网络(Generative Adversarial Networks, GANs)**:由两个网络组成,一个生成器和一个判别器,它们相互竞争,生成器生成数据,判别器评估数据的真实性。 7. **深度学习框架**:如 TensorFlow、Keras、PyTorch 等,这些框架提供了构建、训练和部署深度学习模型的工具和库。 8. **激活函数(Activation Functions)**:如 ReLU、Sigmoid、Tanh 等,它们在神经网络中用于添加非线性,使得网络能够学习复杂的函数。 9. **损失函数(Loss Functions)**:用于评估模型的预测与真实值之间的差异,常见的损失函数包括均方误差(MSE)、交叉熵(Cross-Entropy)等。 10. **优化算法(Optimization Algorithms)**:如梯度下降(Gradient Descent)、随机梯度下降(SGD)、Adam 等,用于更新网络权重,以最小化损失函数。 11. **正则化(Regularization)**:技术如 Dropout、L1/L2 正则化等,用于防止模型过拟合。 12. **迁移学习(Transfer Learning)**:利用在一个任务上训练好的模型来提高另一个相关任务的性能。 深度学习在许多领域都取得了显著的成就,但它也面临着一些挑战,如对大量数据的依赖、模型的解释性差、计算资源消耗大等。研究人员正在不断探索新的方法来解决这些问题。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值