Запись изображений через URLConnection - PullRequest
3 голосов
/ 03 февраля 2011

Я пытаюсь записать изображение через соединение HttpURLC.

Я знаю, как писать текст, но у меня возникают реальные проблемы при попытке написать изображение

Мне удалось записать на локальный HD с помощью ImageIO:

Но я пытаюсь записать Image by ImageIO по URL и не удалось

URL url = new URL(uploadURL);
connection = (HttpURLConnection) url.openConnection();
connection.setDoOutput(true);
connection.setRequestMethod("POST");
connection.setDoInput(true);
connection.setUseCaches(false);
connection.setRequestProperty("Content-Type", "multipart/form-data;
                                            boundary=" + boundary);
output = new DataOutputStream(connection.getOutputStream());
output.writeBytes("--" + boundary + "\r\n");
output.writeBytes("Content-Disposition: form-data; name=\"" + FIELD_NAME + "\";
                                            filename=\"" + fileName + "\"\r\n");
output.writeBytes("Content-Type: " + dataMimeType + "\r\n");
output.writeBytes("Content-Transfer-Encoding: binary\r\n\r\n");
ImageIO.write(image, imageType, output);

uploadURL - это URL-адрес страницы asp на сервере, которая будет загружать изображение с именем файла, указанным в "content-Disposition: part.

теперь, когда я отправляю эту страницу asp, найдите запрос и найдите имя файла. но не находит файл для загрузки.

Проблема заключается в том, что при записи ImageIO по URL-адресу какое имя файла, на котором пишет ImageIO, будет

Поэтому, пожалуйста, помогите мне, как ImageIO напишет изображение на URLConnection и как я могу узнать имя файла, который я должен использовать на странице asp для загрузки файла

Спасибо, что нашли время, чтобы прочитать этот пост Дилип Агарвал

Ответы [ 2 ]

4 голосов
/ 03 февраля 2011

Сначала я считаю, что вы должны позвонить io.flush(), а затем io.close() после написания изображения.

Второй тип контента мне кажется странным. Кажется, что вы пытаетесь отправить форму, в то время как это на самом деле изображение. Я не знаю, чего ожидает ваш asp, но обычно, когда я пишу код, который должен передавать файл по HTTP, я отправляю соответствующий тип контента, например, image/jpeg.

Вот, например, фрагмент кода, который я извлек из одной небольшой утилиты, которую я написал и использую во время моей текущей работы:

    URL url = new URL("http://localhost:8080/handler");
    HttpURLConnection con = (HttpURLConnection)url.openConnection();
    con.setDoInput(true);
    con.setDoOutput(true);
    con.setUseCaches(false);
    con.setRequestProperty("Content-Type", "image/jpeg");
    con.setRequestMethod("POST");
    InputStream in = new FileInputStream("c:/temp/poc/img/mytest2.jpg");
    OutputStream out = con.getOutputStream();
    copy(in, con.getOutputStream());
    out.flush();
    out.close();
    BufferedReader r = new BufferedReader(new  InputStreamReader(con.getInputStream()));


            // obviously it is not required to print the response. But you have
            // to call con.getInputStream(). The connection is really established only
            // when getInputStream() is called.
    System.out.println("Output:");
    for (String line = r.readLine(); line != null;  line = r.readLine()) {
        System.out.println(line);
    }

Я использовал здесь метод copy (), который я взял из утилиты ввода-вывода Jakarta. Вот код для справки:

protected static long copy(InputStream input, OutputStream output)
        throws IOException {
    byte[] buffer = new byte[12288]; // 12K
    long count = 0L;
    int n = 0;
    while (-1 != (n = input.read(buffer))) {
        output.write(buffer, 0, n);
        count += n;
    }
    return count;
}

Очевидно, что сторона сервера должна быть готова к считыванию содержимого изображения непосредственно из тела POST. Надеюсь, это поможет.

0 голосов
/ 23 сентября 2013

ОП кажется заблудшим, но в пользу Мистера Кайта:

// main method 
URL url = new URL(uploadURL);
connection = (HttpURLConnection) url.openConnection();
connection.setDoOutput(true); // triggers "POST"
// connection.setDoInput(true); // only if needed
connection.setUseCaches(false); // dunno
final String boundary = Long.toHexString(System.currentTimeMillis());
connection.setRequestProperty("Content-Type", "multipart/form-data; boundary="
                                                                + boundary);
output = new DataOutputStream(connection.getOutputStream());
try {
     // image must be a File instance
    flushMultiPartData(image, output, boundary);
} catch (IOException e) {
    System.out.println("IOException in flushMultiPartData : " + e);
    return;
}
// ...
private void flushMultiPartData(File file, OutputStream serverOutputStream,
            String boundary) throws FileNotFoundException, IOException {
    // SEE https://stackoverflow.com/a/2793153/281545
    PrintWriter writer = null;
    try {
        // true = autoFlush, important!
        writer = new PrintWriter(new OutputStreamWriter(serverOutputStream,
                charsetForMultipartHeaders), true);
        appendBinary(file, boundary, writer, serverOutputStream);
        // End of multipart/form-data.
        writer.append("--" + boundary + "--").append(CRLF);
    } finally {
        if (writer != null) writer.close();
    }
}

private void appendBinary(File file, String boundary, PrintWriter writer,
        OutputStream output) throws FileNotFoundException, IOException {
    // Send binary file.
    writer.append("--" + boundary).append(CRLF);
    writer.append(
        "Content-Disposition: form-data; name=\"binaryFile\"; filename=\""
            + file.getName() + "\"").append(CRLF);
    writer.append("Content-Type: "
            +  URLConnection.guessContentTypeFromName(file.getName()))
            .append(CRLF);
    writer.append("Content-Transfer-Encoding: binary").append(CRLF);
    writer.append(CRLF).flush();
    InputStream input = null;
    try {
        input = new FileInputStream(file);
        byte[] buffer = new byte[1024];
        for (int length = 0; (length = input.read(buffer)) > 0;) {
            output.write(buffer, 0, length);
        }
        output.flush(); // Important! Output cannot be closed. Close of
        // writer will close output as well.
    } finally {
        if (input != null) try {
            input.close();
        } catch (IOException logOrIgnore) {}
    }
    writer.append(CRLF).flush(); // CRLF is important! It indicates end of
    // binary boundary.
}

Возможно, вы захотите добавить сжатие Gzip - см. Файл , поврежденный, когда я отправляю его в сервлет с помощью GZIPOutputStream для рабочего класса с Gzip или без него. В ImageIO здесь нет места - просто запишите байты за провод и используйте ImageIO для своего сердца на сервере. На основании @BalusC ответа

...