Heim  >  Artikel  >  Wie lade ich Dateien mit JSP/Servlet auf den Server hoch?

Wie lade ich Dateien mit JSP/Servlet auf den Server hoch?

王林
王林nach vorne
2024-02-22 12:40:07797Durchsuche

Der PHP-Editor Youzi zeigt Ihnen, wie Sie mit JSP/Servlet Dateien in Java auf den Server hochladen. In der Webentwicklung ist das Hochladen von Dateien eine häufige Anforderung, und JSP/Servlet kann zur Implementierung einer einfachen und effizienten Funktion zum Hochladen von Dateien verwendet werden. Als Nächstes werden wir die Schritte zum Hochladen von Dateien auf den Server über Java-Code im Detail vorstellen, damit Sie diese Fähigkeit schnell beherrschen und Ihre Entwicklungsfähigkeiten verbessern können.

Frageninhalt

Wie verwende ich JSP/Servlet, um Dateien auf den Server hochzuladen?

Ich habe Folgendes versucht:

<form action="upload" method="post">
    <input type="text" name="description" />
    <input type="file" name="file" />
    <input type="submit" />
</form>

Allerdings erhalte ich nur den Dateinamen, nicht den Dateiinhalt. Wann ich werde enctype="multipart/form-data" 添加到 ff9c23ada1bcecdd1a0fb5d5a0f18437 时,request.getparameter() 返回 null.

Während meiner Recherche bin ich auf den gemeinsamen Dateiupload von Apache gestoßen. Ich habe Folgendes versucht:

fileitemfactory factory = new diskfileitemfactory();
servletfileupload upload = new servletfileupload(factory);
list items = upload.parserequest(request); // this line is where it died.

Leider hat das Servlet eine Ausnahme ohne klare Nachricht oder Begründung ausgelöst. Hier ist der Stacktrace:

SEVERE: Servlet.service() for servlet UploadServlet threw exception
javax.servlet.ServletException: Servlet execution threw an exception
    at org.apache.catalina.core.ApplicationFilterChain.internalDoFilter(ApplicationFilterChain.java:313)
    at org.apache.catalina.core.ApplicationFilterChain.doFilter(ApplicationFilterChain.java:206)
    at org.apache.catalina.core.StandardWrapperValve.invoke(StandardWrapperValve.java:233)
    at org.apache.catalina.core.StandardContextValve.invoke(StandardContextValve.java:191)
    at org.apache.catalina.core.StandardHostValve.invoke(StandardHostValve.java:127)
    at org.apache.catalina.valves.ErrorReportValve.invoke(ErrorReportValve.java:102)
    at org.apache.catalina.core.StandardEngineValve.invoke(StandardEngineValve.java:109)
    at org.apache.catalina.connector.CoyoteAdapter.service(CoyoteAdapter.java:298)
    at org.apache.coyote.http11.Http11Processor.process(Http11Processor.java:852)
    at org.apache.coyote.http11.Http11Protocol$Http11ConnectionHandler.process(Http11Protocol.java:588)
    at org.apache.tomcat.util.net.JIoEndpoint$Worker.run(JIoEndpoint.java:489)
    at java.lang.Thread.run(Thread.java:637)

Lösung

Einführung

Um Dateien zum Hochladen zu durchsuchen und auszuwählen, müssen Sie dem Formular HTML hinzufügen a2dc5349fb8bb852eaec4b6390c03b14 字段。如 HTML specification 中所述,您必须使用 post 方法,并且表单的 enctype 属性必须设置为 "multipart/form-data".

<form action="upload" method="post" enctype="multipart/form-data">
    <input type="text" name="description" />
    <input type="file" name="file" />
    <input type="submit" />
</form>

Nach dem Absenden eines solchen Formulars stehen binäre mehrteilige Formulardaten im Anforderungstext in enctypeeinem anderen Format zur Verfügung, als wenn nicht festgelegt ist.

Vor Servlet 3.0 (Dezember 2009) unterstützte die Servlet-API selbst nicht multipart/form-data。它仅支持默认形式 enctype application/x-www-form-urlencoded。当使用多部分表单数据时,request.getparameter() 和 consorts 都将返回 null. Hier kommt der bekannte Apache Commons FileUpload ins Spiel.

Parsen Sie es nicht manuell!

Theoretisch können Sie den Anfragetext basierend auf ServletRequest#getInputStream() selbst analysieren. Dies ist jedoch eine präzise und mühsame Aufgabe, die genaue Kenntnisse von RFC2388 erfordert. Sie sollten nicht versuchen, dies selbst zu tun oder selbstgemachten, bibliothekslosen Code, den Sie an anderer Stelle im Internet finden, kopieren und einfügen. Viele Online-Ressourcen versagen in dieser Hinsicht, beispielsweise roseindia.net. Siehe auch Hochladen einer PDF-Datei. Sie sollten eine echte Bibliothek verwenden, die seit Jahren von Millionen Benutzern verwendet (und implizit getestet!) wird. Eine solche Bibliothek hat ihre Robustheit bewiesen.

Wenn Sie bereits Servlet 3.0 oder höher verwenden, verwenden Sie bitte die native API

Wenn Sie mindestens Servlet 3.0 verwenden (Tomcat 7, Jetty 9, JBoss As 6, Glassfish 3 usw., es gibt sie seit 2010), dann können Sie HttpServletRequest#getPart() verwenden HttpServletRequest#getPart() 提供的标准 api 来收集单独的多部分表单数据项(大多数 servlet 3.0 实现实际上都在幕后使用 apache commons fileupload!)。此外,正常形式字段可以通过 getparameter() Bietet eine Standard-API zum Sammeln einzelner mehrteiliger Formulardatenelemente (die meisten Servlet 3.0-Implementierungen verwenden tatsächlich Apache Commons FileUpload hinter den Kulissen!). Darüber hinaus können normale Formularfelder wie gewohnt über getparameter() abgerufen werden.

Annotieren Sie Ihr Servlet zunächst mit @MultipartConfig@MultipartConfig 注释您的 servlet,以便让它识别并支持 multipart/form-data 请求,从而使 getpart(), damit es multipart/form-data-Anfragen erkennt und unterstützt, sodass getpart() funktioniert normalerweise:

@webservlet("/upload")
@multipartconfig
public class uploadservlet extends httpservlet {
    // ...
}

Anschließend setzen Sie es dopost() wie folgt um:

protected void dopost(httpservletrequest request, httpservletresponse response) throws servletexception, ioexception {
    string description = request.getparameter("description"); // retrieves <input type="text" name="description">
    part filepart = request.getpart("file"); // retrieves <input type="file" name="file">
    string filename = paths.get(filepart.getsubmittedfilename()).getfilename().tostring(); // msie fix.
    inputstream filecontent = filepart.getinputstream();
    // ... (do your job here)
}

Achtung path#getfilename(). Dies ist ein Msie-Fix bezüglich des Abrufens von Dateinamen. Dieser Browser sendet fälschlicherweise den vollständigen Dateipfad und -namen und nicht nur den Dateinamen.

Wenn Sie mehrere Dateien über multiple="true",

hochladen möchten
<input type="file" name="files" multiple="true" />

Oder auf die altmodische Art und Weise mit mehreren Eingaben,

<input type="file" name="files" />
<input type="file" name="files" />
<input type="file" name="files" />
...

Sie können sie dann wie folgt sammeln (eine solche Methode wie request.getparts("files") gibt es leider nicht):

protected void dopost(httpservletrequest request, httpservletresponse response) throws servletexception, ioexception {
    // ...
    list<part> fileparts = request.getparts().stream().filter(part -> "files".equals(part.getname()) && part.getsize() > 0).collect(collectors.tolist()); // retrieves <input type="file" name="files" multiple="true">

    for (part filepart : fileparts) {
        string filename = paths.get(filepart.getsubmittedfilename()).getfilename().tostring(); // msie fix.
        inputstream filecontent = filepart.getinputstream();
        // ... (do your job here)
    }
}

当您尚未使用 servlet 3.1 时,手动获取提交的文件名

请注意,Part#getSubmittedFileName() 是在 servlet 3.1 中引入的(tomcat 8、jetty 9、wildfly 8、glassfish 4 等,它们自 2013 年以来就已存在)。如果您还没有使用 servlet 3.1(真的吗?),那么您需要一个额外的实用方法来获取提交的文件名。

private static string getsubmittedfilename(part part) {
    for (string cd : part.getheader("content-disposition").split(";")) {
        if (cd.trim().startswith("filename")) {
            string filename = cd.substring(cd.indexof('=') + 1).trim().replace("\"", "");
            return filename.substring(filename.lastindexof('/') + 1).substring(filename.lastindexof('\\') + 1); // msie fix.
        }
    }
    return null;
}
string filename = getsubmittedfilename(filepart);

请注意有关获取文件名的 msie 修复。此浏览器错误地发送完整文件路径和名称,而不仅仅是文件名。

当您尚未使用 servlet 3.0 时,请使用 apache commons fileupload

如果您还没有使用 servlet 3.0(是不是该升级了?它已经十多年前发布了!),通常的做法是使用 Apache Commons FileUpload 来解析多部分表单数据请求。它有一个很好的User GuideFAQ(仔细检查两者)。还有 o'reilly(“cos”)multipartrequest,但它有一些(小)错误,并且多年来不再积极维护。我不建议使用它。 apache commons fileupload 仍在积极维护,目前非常成熟。

为了使用 apache commons fileupload,您的 web 应用程序的 /web-inf/lib 中至少需要有以下文件:

您最初的尝试失败很可能是因为您忘记了公共 io。

下面是一个启动示例,显示使用 apache commons fileupload 时 uploadservletdopost() 的样子:

protected void dopost(httpservletrequest request, httpservletresponse response) throws servletexception, ioexception {
    try {
        list<fileitem> items = new servletfileupload(new diskfileitemfactory()).parserequest(request);
        for (fileitem item : items) {
            if (item.isformfield()) {
                // process regular form field (input type="text|radio|checkbox|etc", select, etc).
                string fieldname = item.getfieldname();
                string fieldvalue = item.getstring();
                // ... (do your job here)
            } else {
                // process form file field (input type="file").
                string fieldname = item.getfieldname();
                string filename = filenameutils.getname(item.getname());
                inputstream filecontent = item.getinputstream();
                // ... (do your job here)
            }
        }
    } catch (fileuploadexception e) {
        throw new servletexception("cannot parse multipart request.", e);
    }

    // ...
}

非常重要的是,您不要事先在同一请求上调用 getparameter()getparametermap()getparametervalues()getinputstream()getreader() 等。否则,servlet 容器将读取并解析请求正文,因此 apache commons fileupload 将得到一个空请求正文。另请参见 a.o. ServletFileUpload#parseRequest(request) returns an empty list

注意 filenameutils#getname()。这是有关获取文件名的 msie 修复。此浏览器错误地发送完整文件路径和名称,而不仅仅是文件名。

或者,您也可以将这一切包装在 filter 中,它会自动解析所有内容并将内容放回请求的参数映射中,以便您可以继续以通常的方式使用 request.getparameter() 并通过以下方式检索上传的文件request.getattribute()You can find an example in this blog article

glassfish3 中 getparameter() 错误的解决方法仍然返回 null

请注意,早于 3.1.2 的 glassfish 版本有 a bug,其中 getparameter() 仍返回 null。如果您的目标是这样的容器并且无法升级它,那么您需要借助此实用程序方法从 getpart() 中提取值:

private static string getvalue(part part) throws ioexception {
    bufferedreader reader = new bufferedreader(new inputstreamreader(part.getinputstream(), "utf-8"));
    stringbuilder value = new stringbuilder();
    char[] buffer = new char[1024];
    for (int length = 0; (length = reader.read(buffer)) > 0;) {
        value.append(buffer, 0, length);
    }
    return value.tostring();
}
string description = getvalue(request.getpart("description")); // retrieves <input type="text" name="description">

保存上传的文件(不要使用 getrealpath()part.write()!)

前往以下答案,详细了解如何将获得的 inputstream (上述代码片段中所示的 filecontent 变量)正确保存到磁盘或数据库:

提供上传的文件

请参阅以下答案,了解如何正确地将保存的文件从磁盘或数据库返回给客户端的详细信息:

ajax 表单

前往以下答案,了解如何使用 ajax(和 jquery)上传。请注意,无需为此更改用于收集表单数据的 servlet 代码!只有您响应的方式可能会改变,但这相当简单(即,不转发到 jsp,只需打印一些 json 或 xml 甚至纯文本,具体取决于负责 ajax 调用的脚本所期望的内容)。

如果您碰巧使用 Spring MVC,请按以下方法操作(我将其留在这里,以防有人发现它有用):

使用 enctype 属性设置为“multipart/form-datazqbenczqb”的表单(与 <a href="https://www.php.cn/link/8a13dab3f5ec9e22d0d1495c8c85e436">BalusC's answer</a> 相同):94b3e26ee717c64999d7867364b1b4a3 <pre class="brush:html;toolbar:false;">&lt;form action=&quot;upload&quot; method=&quot;post&quot; enctype=&quot;multipart/form-data&quot;&gt; &lt;input type=&quot;file&quot; name=&quot;file&quot; /&gt; &lt;input type=&quot;submit&quot; value=&quot;upload&quot;/&gt; &lt;/form&gt; </pre> e388a4556c0f65e1904146cc1a846bee在您的控制器中,将请求参数 <code>file 映射到 multipartfile 类型,如下所示:

@RequestMapping(value = "/upload", method = RequestMethod.POST)
public void handleUpload(@RequestParam("file") MultipartFile file) throws IOException {
    if (!file.isEmpty()) {
            byte[] bytes = file.getBytes(); // alternatively, file.getInputStream();
            // application logic
    }
}

您可以使用 multipartfilegetoriginalfilename()getsize() 获取文件名和大小。

我已经使用 spring 版本 4.1.1.release 对此进行了测试。

Das obige ist der detaillierte Inhalt vonWie lade ich Dateien mit JSP/Servlet auf den Server hoch?. Für weitere Informationen folgen Sie bitte anderen verwandten Artikeln auf der PHP chinesischen Website!

Stellungnahme:
Dieser Artikel ist reproduziert unter:stackoverflow.com. Bei Verstößen wenden Sie sich bitte an admin@php.cn löschen