Надсилання запиту HTTP POST в Java


294

дозволимо припустити цю URL-адресу ...

http://www.example.com/page.php?id=10            

(Тут ідентифікатор потрібно надіслати у запиті POST)

Я хочу надіслати на id = 10сервер page.php, який приймає його методом POST.

Як я можу це зробити з Java?

Я спробував це:

URL aaa = new URL("http://www.example.com/page.php");
URLConnection ccc = aaa.openConnection();

Але я все ще не можу зрозуміти, як надіслати його через POST

Відповіді:


339

Оновлений відповідь:

Оскільки деякі класи в початковій відповіді застаріли в новій версії компонентів HTTP HTTP Apache, я публікую це оновлення.

До речі, ви можете отримати повну документацію, щоб отримати більше прикладів тут .

HttpClient httpclient = HttpClients.createDefault();
HttpPost httppost = new HttpPost("http://www.a-domain.com/foo/");

// Request parameters and other properties.
List<NameValuePair> params = new ArrayList<NameValuePair>(2);
params.add(new BasicNameValuePair("param-1", "12345"));
params.add(new BasicNameValuePair("param-2", "Hello!"));
httppost.setEntity(new UrlEncodedFormEntity(params, "UTF-8"));

//Execute and get the response.
HttpResponse response = httpclient.execute(httppost);
HttpEntity entity = response.getEntity();

if (entity != null) {
    try (InputStream instream = entity.getContent()) {
        // do something useful
    }
}

Оригінальний відповідь:

Рекомендую використовувати Apache HttpClient. його швидше і простіше у виконанні.

HttpPost post = new HttpPost("http://jakarata.apache.org/");
NameValuePair[] data = {
    new NameValuePair("user", "joe"),
    new NameValuePair("password", "bloggs")
};
post.setRequestBody(data);
// execute method and handle any error responses.
...
InputStream in = post.getResponseBodyAsStream();
// handle response.

Для отримання додаткової інформації ознайомтеся з цією URL-адресою: http://hc.apache.org/


25
Після деякого часу, коли я намагався взяти на себе руки, PostMethodздається, що насправді зараз це називається HttpPostзгідно stackoverflow.com/a/9242394/1338936 - просто для тих, хто знайшов цю відповідь, як я :)
Мартін Лін

1
@Juan (та Мартін Лін) дякую за коментарі. Я щойно оновив відповідь.
mhshams

Чи використовується ваша переглянута відповідь hc.apache.org?
djangofan

@djangofan так. в переглянутій відповіді також є посилання на apache-hc.
mhshams

6
вам слід додати імпортовані libs
gouchaoer

191

Надіслати POST-запит у ванільній Java легко. Починаючи з а URL, нам потрібно перетворити його на URLConnectionвикористання url.openConnection();. Після цього нам потрібно передати це на a HttpURLConnection, щоб ми могли отримати доступ до його setRequestMethod()методу для встановлення нашого методу. Ми нарешті кажемо, що ми будемо надсилати дані через з'єднання.

URL url = new URL("https://www.example.com/login");
URLConnection con = url.openConnection();
HttpURLConnection http = (HttpURLConnection)con;
http.setRequestMethod("POST"); // PUT is another valid option
http.setDoOutput(true);

Потім нам потрібно зазначити, що ми будемо надсилати:

Надіслати просту форму

Звичайний POST, що надходить із форми http, має чітко визначений формат. Нам потрібно конвертувати наш вклад у такий формат:

Map<String,String> arguments = new HashMap<>();
arguments.put("username", "root");
arguments.put("password", "sjh76HSn!"); // This is a fake password obviously
StringJoiner sj = new StringJoiner("&");
for(Map.Entry<String,String> entry : arguments.entrySet())
    sj.add(URLEncoder.encode(entry.getKey(), "UTF-8") + "=" 
         + URLEncoder.encode(entry.getValue(), "UTF-8"));
byte[] out = sj.toString().getBytes(StandardCharsets.UTF_8);
int length = out.length;

Потім ми можемо прикріпити вміст форми до запиту http із відповідними заголовками та надіслати його.

http.setFixedLengthStreamingMode(length);
http.setRequestProperty("Content-Type", "application/x-www-form-urlencoded; charset=UTF-8");
http.connect();
try(OutputStream os = http.getOutputStream()) {
    os.write(out);
}
// Do something with http.getInputStream()

Надсилання JSON

Ми також можемо надсилати json за допомогою Java, це також просто:

byte[] out = "{\"username\":\"root\",\"password\":\"password\"}" .getBytes(StandardCharsets.UTF_8);
int length = out.length;

http.setFixedLengthStreamingMode(length);
http.setRequestProperty("Content-Type", "application/json; charset=UTF-8");
http.connect();
try(OutputStream os = http.getOutputStream()) {
    os.write(out);
}
// Do something with http.getInputStream()

Пам'ятайте, що різні сервери приймають різні типи вмісту для json, дивіться це питання.


Надсилання файлів з повідомленням Java

Надсилання файлів можна вважати складнішими для обробки, оскільки формат складніший. Ми також будемо додавати підтримку для надсилання файлів у вигляді рядка, оскільки ми не хочемо повністю завантажувати файл у пам'ять.

Для цього ми визначаємо кілька допоміжних методів:

private void sendFile(OutputStream out, String name, InputStream in, String fileName) {
    String o = "Content-Disposition: form-data; name=\"" + URLEncoder.encode(name,"UTF-8") 
             + "\"; filename=\"" + URLEncoder.encode(filename,"UTF-8") + "\"\r\n\r\n";
    out.write(o.getBytes(StandardCharsets.UTF_8));
    byte[] buffer = new byte[2048];
    for (int n = 0; n >= 0; n = in.read(buffer))
        out.write(buffer, 0, n);
    out.write("\r\n".getBytes(StandardCharsets.UTF_8));
}

private void sendField(OutputStream out, String name, String field) {
    String o = "Content-Disposition: form-data; name=\"" 
             + URLEncoder.encode(name,"UTF-8") + "\"\r\n\r\n";
    out.write(o.getBytes(StandardCharsets.UTF_8));
    out.write(URLEncoder.encode(field,"UTF-8").getBytes(StandardCharsets.UTF_8));
    out.write("\r\n".getBytes(StandardCharsets.UTF_8));
}

Тоді ми можемо використовувати ці методи для створення запиту про багатостороннє повідомлення наступним чином:

String boundary = UUID.randomUUID().toString();
byte[] boundaryBytes = 
           ("--" + boundary + "\r\n").getBytes(StandardCharsets.UTF_8);
byte[] finishBoundaryBytes = 
           ("--" + boundary + "--").getBytes(StandardCharsets.UTF_8);
http.setRequestProperty("Content-Type", 
           "multipart/form-data; charset=UTF-8; boundary=" + boundary);

// Enable streaming mode with default settings
http.setChunkedStreamingMode(0); 

// Send our fields:
try(OutputStream out = http.getOutputStream()) {
    // Send our header (thx Algoman)
    out.write(boundaryBytes);

    // Send our first field
    sendField(out, "username", "root");

    // Send a seperator
    out.write(boundaryBytes);

    // Send our second field
    sendField(out, "password", "toor");

    // Send another seperator
    out.write(boundaryBytes);

    // Send our file
    try(InputStream file = new FileInputStream("test.txt")) {
        sendFile(out, "identification", file, "text.txt");
    }

    // Finish the request
    out.write(finishBoundaryBytes);
}


// Do something with http.getInputStream()

5
Ця публікація корисна, але досить хибна. Мені знадобилося 2 дні, щоб я працював. Тому для його роботи потрібно замінити StandartCharsets.UTF8 на StandardCharsets.UTF_8. borderaryBytes та закінченняBoundaryBytes повинні отримати дві додаткові дефіси, які НЕ передаються у Content-Type, тому borderaryBytes = ("-" + межа + "\ r \ n"). Отримати ... Вам також потрібно передати borderBytes один раз ДО ПЕРШОГО першого або першого поля буде ігноровано!
Алгоман

Навіщо out.write(finishBoundaryBytes);потрібна лінія? http.connect();виконає надсилання POST, чи не так?
János

16
"Надіслати POST-запит у ванільній Java легко". А далі слідують десятки рядків коду, порівняно з чимось на зразок requests.post('http://httpbin.org/post', data = {'key':'value'})Python… Я новачок у Java, тому для мене це дуже дивне використання слова “легко” :)
Лінн

1
Це відносно простіше, ніж те, що я очікував, вважаючи, що це Java :)
shaahiin

загадковий \ r \ n \ r \ n означає CRLF CRLF (повернення каретки + лінія каналу). Це створює 2х нову лінію. Перший новий рядок - це закінчити поточну лінію. Другий рядок полягає в тому, щоб відрізняти заголовки http від http body в запиті. HTTP - протокол на основі ASCII. Це правило для вставки \ r \ n
Мітья Густін

99
String rawData = "id=10";
String type = "application/x-www-form-urlencoded";
String encodedData = URLEncoder.encode( rawData, "UTF-8" ); 
URL u = new URL("http://www.example.com/page.php");
HttpURLConnection conn = (HttpURLConnection) u.openConnection();
conn.setDoOutput(true);
conn.setRequestMethod("POST");
conn.setRequestProperty( "Content-Type", type );
conn.setRequestProperty( "Content-Length", String.valueOf(encodedData.length()));
OutputStream os = conn.getOutputStream();
os.write(encodedData.getBytes());

Важливо зауважити: використання нічого іншого, ніж String.getBytes (), здається, не працює. Наприклад, використання PrintWriter повністю не вдається.
Столики маленького бобі

5
і як встановити 2 публікації даних? Розділити двокрапкою, комою?
галасливий кіт

10
encode(String)застаріло. Вам потрібно скористатися encode(String, String), яке вказує тип кодування. Приклад: encode(rawData, "UTF-8").
судо

3
Ви можете виконати наступне в кінці. Це дозволить переконатися, що запит закінчено, і сервер отримує можливість обробити відповідь: conn.getResponseCode ();
Шимон Яхім

3
не кодуйте весь рядок .. ви повинні кодувати лише значення кожного параметра
user2914191

22

Перша відповідь була чудовою, але мені довелося додати спробувати / catch, щоб уникнути помилок компілятора Java.
Крім того, у мене виникли проблеми з розумінням того, як читати HttpResponseз бібліотеками Java.

Ось більш повний код:

/*
 * Create the POST request
 */
HttpClient httpClient = new DefaultHttpClient();
HttpPost httpPost = new HttpPost("http://example.com/");
// Request parameters and other properties.
List<NameValuePair> params = new ArrayList<NameValuePair>();
params.add(new BasicNameValuePair("user", "Bob"));
try {
    httpPost.setEntity(new UrlEncodedFormEntity(params, "UTF-8"));
} catch (UnsupportedEncodingException e) {
    // writing error to Log
    e.printStackTrace();
}
/*
 * Execute the HTTP Request
 */
try {
    HttpResponse response = httpClient.execute(httpPost);
    HttpEntity respEntity = response.getEntity();

    if (respEntity != null) {
        // EntityUtils to get the response content
        String content =  EntityUtils.toString(respEntity);
    }
} catch (ClientProtocolException e) {
    // writing exception to log
    e.printStackTrace();
} catch (IOException e) {
    // writing exception to log
    e.printStackTrace();
}

EntityUtils був корисним.
Джей

6
Вибачте, але ви не вловили жодних помилок, ви ввели їх. Ловити винятки в тому місці, де ви не можете впоратися з ними, - це неправильно і e.printStackTrace()нічого не вдається.
maaartinus

java.net.ConnectException: Час з'єднання вичерпано: підключення
kerZy Hart

17

Простий спосіб використання HTTP-компонентів Apache - це

Request.Post("http://www.example.com/page.php")
            .bodyForm(Form.form().add("id", "10").build())
            .execute()
            .returnContent();

Погляньте на API Fluent


3
Просто для зручності; налаштування залежностей / info: hc.apache.org/httpcomponents-client-4.5.x/httpclient/… та hc.apache.org/httpcomponents-client-4.5.x/fluent-hc/…
Ярослав Заруба

5

найпростіший спосіб надсилання параметрів із запитом на пошту:

String postURL = "http://www.example.com/page.php";

HttpPost post = new HttpPost(postURL);

List<NameValuePair> params = new ArrayList<NameValuePair>();
params.add(new BasicNameValuePair("id", "10"));

UrlEncodedFormEntity ent = new UrlEncodedFormEntity(params, "UTF-8");
post.setEntity(ent);

HttpClient client = new DefaultHttpClient();
HttpResponse responsePOST = client.execute(post);

Ви зробили. тепер ви можете використовувати responsePOST. Отримайте вміст відповіді як рядок:

BufferedReader reader = new BufferedReader(new  InputStreamReader(responsePOST.getEntity().getContent()), 2048);

if (responsePOST != null) {
    StringBuilder sb = new StringBuilder();
    String line;
    while ((line = reader.readLine()) != null) {
        System.out.println(" line : " + line);
        sb.append(line);
    }
    String getResponseString = "";
    getResponseString = sb.toString();
//use server output getResponseString as string value.
}

1

Виклик HttpURLConnection.setRequestMethod("POST")і HttpURLConnection.setDoOutput(true);насправді потрібен лише останній, оскільки POST потім стає методом за замовчуванням.


it it HttpURLConnection.setRequestMethod () :)
Хосе Діаз

1

Я рекомендую використовувати http-запит, побудований на апачі http api.

HttpRequest<String> httpRequest = HttpRequestBuilder.createPost("http://www.example.com/page.php", String.class)
.responseDeserializer(ResponseDeserializer.ignorableDeserializer()).build();

public void send(){
   String response = httpRequest.execute("id", "10").get();
}
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.