Java 中的 Http POST(带文件上传)

2022-09-03 12:46:48

我想做的是从Java应用程序提交一个Web表单。我需要填写的表格位于此处:http://cando-dna-origami.org/

提交表单后,服务器会向给定的电子邮件地址发送一封确认电子邮件,目前我只是手动检查。我尝试手动填写表格,电子邮件发送正常。(还应该注意的是,当表单填写不正确时,页面只会刷新,不会提供任何反馈)。

我以前从未用http做过任何事情,但是我环顾四周,并想出了以下代码,该代码应该向服务器发送POST请求:

    String data = "name=M+V&affiliation=Company&email="
            + URLEncoder.encode("m.v@gmail.com", "UTF-8")
            + "&axialRise=0.34&helixDiameter=2.25&axialStiffness=1100&bendingStiffness=230" +
            "&torsionalStiffness=460&nickStiffness=0.01&resolution=course&jsonUpload="
            + URLEncoder.encode("C:/Users/Marjie/Downloads/twisted_DNA_bundles/monotwist.L1.v1.json",
            "UTF-8") + "&type=square";

    URL page = new URL("http://cando-dna-origami.org/");
    HttpURLConnection con = (HttpURLConnection) page.openConnection();

    con.setDoOutput(true);
    con.setRequestMethod("POST");
    con.connect();

    OutputStreamWriter out = new OutputStreamWriter(con.getOutputStream());
    out.write(data);
    out.flush();

    System.out.println(con.getResponseCode());
    System.out.println(con.getResponseMessage());

    out.close();
    con.disconnect();

但是,当它运行时,它似乎没有任何作用 - 也就是说,我没有收到任何电子邮件,尽管该程序确实将“200 OK”打印到System.out,这似乎表明从服务器收到了某些东西,尽管我不确定它到底是什么意思。我认为问题可能出在文件上传上,因为我不确定该数据类型是否需要不同的格式。

这是使用Java发送POST请求的正确方法吗?我是否需要为文件上传执行其他操作?谢谢!


在阅读了Adam的帖子后,我使用了Apache HttpClient并编写了以下代码:

    List<NameValuePair> params = new ArrayList<NameValuePair>();
    params.add(new BasicNameValuePair("type", "square"));
    //... add more parameters

    UrlEncodedFormEntity entity = new UrlEncodedFormEntity(params, HTTP.UTF_8);

    HttpPost post = new HttpPost("http://cando-dna-origami.org/");
    post.setEntity(entity);

    HttpResponse response = new DefaultHttpClient().execute(post);
    post = new HttpPost("http://cando-dna-origami.org/");

    post.setEntity(new FileEntity(new File("C:/Users/Marjie/Downloads/twisted_DNA_bundles/monotwist.L1.v1.json"), "text/plain; charset=\"UTF-8\""));
    HttpResponse responseTwo = new DefaultHttpClient().execute(post);

但是,它似乎仍然不起作用。同样,我不确定上传的文件如何适应表单,所以我尝试只发送两个单独的POST请求,一个带有表单,另一个包含其他数据。我仍在寻找一种方法将这些组合成一个请求;有人知道吗?


答案 1

使用像Apache HttpClient这样的东西可能会更好,通过它,你可以以编程方式构建请求。POST

HttpClient httpclient = new DefaultHttpClient();
HttpPost httppost = new HttpPost("http://.../whatever");

List <NameValuePair> params = new ArrayList<NameValuePair>();
params.add(new BasicNameValuePair("param1", "value1"));
params.add(new BasicNameValuePair("param2", "value2"));
...

httpost.setEntity(new UrlEncodedFormEntity(params, HTTP.UTF_8));

HttpResponse response = httpclient.execute(httppost);

如果您需要将文件与表单一起上传,则需要改用:MultipartEntity

MultipartEntity reqEntity = new MultipartEntity();
reqEntity.addPart("someParam", "someValue");
reqEntity.addPart("someFile", new FileBody("/some/file"));
....

httpost.setEntity(reqEntity);

他们的网站上有一些示例程序。“基于表单的登录”和“多部分编码的请求实体”就是很好的例子。

测试您的连接并查看底层网络数据以了解正在发生的事情可能也是值得的。像Firebug这样的东西可以让你准确地看到浏览器中发生了什么,你可以打开HttpClient日志记录来查看程序中交换的所有数据。或者,您可以使用Wireshark或Fiddler之类的东西来实时观察您的网络流量。这可以让您更好地了解浏览器正在做什么,而不是程序在做什么。


答案 2

由于大多数建议的Java HTTP POST请求代码都无法运行,我决定为您提供我的全部操作代码,我相信您将来会发现这有助于创建任何基于Java的POST请求。

此 POST 请求的类型允许将文件发送/上载到服务器。multipart

多部分请求由一个主标头和一个名为 boundary 的分隔符字符串组成,用于将每个部分与另一部分区分开来(此分隔符将在流中以“--”(两个破折号)字符串出现,并且每个部分都有自己的小标头来告诉其类型和更多元数据。

我的任务是使用一些在线服务创建一个PDF文件,但所有多部分POST示例都没有解决问题......

我需要将HTML文档及其图片,JS和CSS文件打包在ZIP / TAR文件中,将其上传到在线html2pdf转换服务,并将结果作为PDF文档作为来自该服务的响应(流)返回给我。

我使用以下代码检查的当前服务是:Htmlpdfapi.com 但我确信,通过微小的调整,您将能够将其与任何其他服务一起使用。

方法调用(对于该服务)如下所示:[class instance name].sendPOSTRequest("http://htmlpdfapi.com/api/v1/pdf", "Token 6hr4-AmqZDrFVjAcJGykjYyXfwG1wER4", "/home/user/project/srv/files/example.zip", "result.pdf");

这是我的代码,经过检查,100%有效:

public void sendPOSTRequest(String url, String authData, String attachmentFilePath, String outputFilePathName)
{
    String charset = "UTF-8";
    File binaryFile = new File(attachmentFilePath);
    String boundary = "------------------------" + Long.toHexString(System.currentTimeMillis()); // Just generate some unique random value.
    String CRLF = "\r\n"; // Line separator required by multipart/form-data.
    int    responseCode = 0;

    try 
    {
        //Set POST general headers along with the boundary string (the seperator string of each part)
        URLConnection connection = new URL(url).openConnection();
        connection.setDoOutput(true);
        connection.setRequestProperty("Content-Type", "multipart/form-data; boundary=" + boundary);
        connection.addRequestProperty("User-Agent", "CheckpaySrv/1.0.0");
        connection.addRequestProperty("Accept", "*/*");
        connection.addRequestProperty("Authentication", authData);

        OutputStream output = connection.getOutputStream();
        PrintWriter writer  = new PrintWriter(new OutputStreamWriter(output, charset), true);

        // Send binary file - part
        // Part header
        writer.append("--" + boundary).append(CRLF);
        writer.append("Content-Disposition: form-data; name=\"file\"; filename=\"" + binaryFile.getName() + "\"").append(CRLF);
        writer.append("Content-Type: application/octet-stream").append(CRLF);// + URLConnection.guessContentTypeFromName(binaryFile.getName())).append(CRLF);
        writer.append(CRLF).flush();

        // File data
        Files.copy(binaryFile.toPath(), output);
        output.flush(); 

        // End of multipart/form-data.
        writer.append(CRLF).append("--" + boundary + "--").flush();

        responseCode = ((HttpURLConnection) connection).getResponseCode();


        if(responseCode !=200) //We operate only on HTTP code 200
            return;

        InputStream Instream = ((HttpURLConnection) connection).getInputStream();

        // Write PDF file 
        BufferedInputStream BISin = new BufferedInputStream(Instream);
        FileOutputStream FOSfile  = new FileOutputStream(outputFilePathName);
        BufferedOutputStream out  = new BufferedOutputStream(FOSfile);

        int i;
        while ((i = BISin.read()) != -1) {
            out.write(i);
        }

        // Cleanup
        out.flush();
        out.close();


    }
    catch(Exception e)
    {
        e.printStackTrace();
    }

}

推荐