Bootstrap

java中发送HTTP/HTTPS的POST请求总结

说明:

网上关于“java中发送HTTP/HTTPS的POST请求”的技术文章非常多,有的可用,有的不可用,很难分辨,而这些又在日常开发中经常用到。所以个人对以前使用的三类进行了总结(均参考了网络资料,亲测可用),希望以后自己或者网友能够用到,如有不妥之处,欢迎讨论。

发送Form表单类型的POST(HTTP)

依赖包:

import org.apache.http.HttpEntity;
import org.apache.http.client.entity.UrlEncodedFormEntity;
import org.apache.http.client.methods.CloseableHttpResponse;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.HttpClients;
import org.apache.http.message.BasicNameValuePair;
import org.apache.http.util.EntityUtils;

import java.util.ArrayList;
import java.util.List;
import java.util.Map;

代码:

    /**
     * 发送form表单类型的Post请求
     *
     * @param url       发送的url
     * @param headerMap 请求头参数集合 key参数名 value为参数值
     * @param bodyMap   请求参数集合 key参数名 value为参数值
     */
    public static String sendFormPost(String url, Map<String, String> headerMap, Map<String, String> bodyMap) {

        //创建post请求对象
        HttpPost post = new HttpPost(url);
        try {
            //创建参数集合
            List<BasicNameValuePair> list = new ArrayList<>();
            //添加参数
            if (bodyMap != null) {
                for (String str : bodyMap.keySet()
                        ) {
                    list.add(new BasicNameValuePair(str, bodyMap.get(str)));
                }
            }
            //把参数放入请求对象,,post发送的参数list,指定格式
            post.setEntity(new UrlEncodedFormEntity(list, Constants.UTF8));

            if (headerMap != null) {
                for (String str : headerMap.keySet()
                        ) {
                    post.addHeader(str, headerMap.get(str));
                }
            }

            CloseableHttpClient client = HttpClients.createDefault();
            //启动执行请求,并获得返回值
            CloseableHttpResponse response = client.execute(post);
            //得到返回的entity对象
            HttpEntity entity = response.getEntity();
            //把实体对象转换为string
            String result = EntityUtils.toString(entity, Constants.UTF8);
            //为防止频繁调用一个接口导致接口爆掉,每次调用完成后停留100毫秒
            Thread.sleep(100);
            //返回内容
            return result;
        } catch (Exception e1) {
            e1.printStackTrace();
            return "";
        }
    }

发送json数据的POST(HTTP)

依赖:

import org.apache.http.HttpEntity;
import org.apache.http.HttpResponse;
import org.apache.http.client.methods.HttpPost;
import org.apache.http.entity.StringEntity;
import org.apache.http.impl.client.CloseableHttpClient;
import org.apache.http.impl.client.HttpClients;
import org.apache.http.message.BasicHeader;
import org.apache.http.protocol.HTTP;
import org.apache.http.util.EntityUtils;

代码:

/**
 * 发送json数据的post请求
 * 不支持https
 *
 * @param url      路径
 * @param sendData 参数(json串)
 * @return 返回调接口返回信息
 */
public static String sendJsonPost(String url, String sendData) {
    String body = "";
    try {
        CloseableHttpClient client = HttpClients.createDefault();

        //创建post方式请求对象
        HttpPost httpPost = new HttpPost(url);

        //装填参数
        StringEntity s = new StringEntity(sendData, Constants.UTF8);
        s.setContentEncoding(new BasicHeader(HTTP.CONTENT_TYPE,
                "application/json"));
        //设置参数到请求对象中
        httpPost.setEntity(s);
        //设置header信息
        httpPost.setHeader("Content-type", "application/json");
        httpPost.setHeader("User-Agent", "Mozilla/4.0 (compatible; MSIE 5.0; Windows NT; DigExt)");

        //执行请求操作,并拿到结果(同步阻塞)
        HttpResponse response = client.execute(httpPost);
        //获取结果实体
        HttpEntity entity = response.getEntity();
        if (entity != null) {
            //按指定编码转换结果实体为String类型
            body = EntityUtils.toString(entity, Constants.UTF8);
        }
        EntityUtils.consume(entity);
        //为防止频繁调用一个接口导致接口爆掉,每次调用完成后停留100毫秒
        Thread.sleep(100);
    } catch (Exception e) {
        log.info("JSON数据发送失败,异常:{}", e.getMessage());
        log.error("异常:", e);
    }
    return body;
}

发送json数据的POST(HTTPS)

依赖:

import javax.net.ssl.*;
import java.io.ByteArrayOutputStream;
import java.io.DataOutputStream;
import java.io.InputStream;
import java.net.URL;
import java.security.cert.CertificateException;
import java.security.cert.X509Certificate;

代码:

private static class TrustAnyTrustManager implements X509TrustManager {
    /**
     * 该方法检查客户端的证书,若不信任该证书则抛出异常。由于我们不需要对客户端进行认证,因此我们只需要执行默认的信任管理器的这
     * 个方法。
     * JSSE中,默认的信任管理器类为TrustManager。
     */
    @Override
    public void checkClientTrusted(X509Certificate[] chain, String authType)
            throws CertificateException {
    }

    /**
     * 该方法检查服务器的证书,若不信任该证书同样抛出异常。通过自己实现该方法,可以使之信任我们指定的任何证书。
     * 在实现该方法时,也可以简单的不做任何处理, 即一个空的函数体,由于不会抛出异常,它就会信任任何证书。(non-Javadoc)
     */
    @Override
    public void checkServerTrusted(X509Certificate[] chain, String authType)
            throws CertificateException {
    }

    /**
     * @return 返回受信任的X509证书数组。
     */
    @Override
    public X509Certificate[] getAcceptedIssuers() {
        return new X509Certificate[]{};
    }
}

private static class TrustAnyHostnameVerifier implements HostnameVerifier {
    @Override
    public boolean verify(String hostname, SSLSession session) {
        return true;
    }
}

/**
 * post方式请求服务器(https协议)
 *
 * @param url     求地址
 * @param content 参数
 * @return
 */
public static String sendJsonToHttpsPost(String url, String content) {
    try {
        /*
         * 类HttpsURLConnection似乎并没有提供方法设置信任管理器。其实,
         * HttpsURLConnection通过SSLSocket来建立与HTTPS的安全连接
         * ,SSLSocket对象是由SSLSocketFactory生成的。
         * HttpsURLConnection提供了方法setSSLSocketFactory
         * (SSLSocketFactory)设置它使用的SSLSocketFactory对象。
         * SSLSocketFactory通过SSLContext对象来获得,在初始化SSLContext对象时,可指定信任管理器对象。
         */
        SSLContext sc = SSLContext.getInstance("SSL");
        sc.init(null, new TrustManager[]{new TrustAnyTrustManager()},
                new java.security.SecureRandom());

        URL console = new URL(url);
        HttpsURLConnection conn = (HttpsURLConnection) console.openConnection();
        conn.setSSLSocketFactory(sc.getSocketFactory());
        conn.setHostnameVerifier(new TrustAnyHostnameVerifier());
        conn.setDoOutput(true);
        // 设置请求头
        conn.setRequestProperty("Content-Type", "application/json;charset=utf-8");
        conn.connect();
        DataOutputStream out = new DataOutputStream(conn.getOutputStream());
        out.write(content.getBytes(Constants.UTF8));
        // 刷新、关闭
        out.flush();
        out.close();
        InputStream is = conn.getInputStream();
        if (is != null) {
            ByteArrayOutputStream outStream = new ByteArrayOutputStream();
            byte[] buffer = new byte[1024];
            int len = 0;
            while ((len = is.read(buffer)) != -1) {
                outStream.write(buffer, 0, len);
            }
            is.close();
            return new String(outStream.toByteArray(), Constants.UTF8);
        }
    } catch (Exception e) {
        log.info("JSON数据发送失败,异常:{}", e.getMessage());
        log.error("异常:", e);
    }
    return null;
}

转发 MultipartFile 的Form表单类型POST请求

依赖:

  <!-- https://mvnrepository.com/artifact/cn.hutool/hutool-all -->
   <dependency>
       <groupId>cn.hutool</groupId>
       <artifactId>hutool-all</artifactId>
       <version>5.8.12</version>
   </dependency>

代码:

import cn.hutool.http.HttpRequest;
import com.alibaba.fastjson.JSON;
import lombok.extern.slf4j.Slf4j;
import org.springframework.web.multipart.MultipartFile;

import java.io.File;
import java.io.IOException;
import java.util.HashMap;
import java.util.Map;


@Slf4j
public class HttpUtil {


    /**
     * 以post方式调用第三方接口,以 form-data 形式  发送 MultipartFile 文件数据
     *
     * @param url              post请求url
     * @param fileParamName    文件参数名称
     * @param multipartFile    文件
     * @param otherFormDataMap 表单中的其他参数
     * @param connectTimeout   连接超时时间
     * @return 返回 接口返回结果
     */
    public static String doPostFormDataFile(String url, String fileParamName, MultipartFile multipartFile, Map<String, String> otherFormDataMap,
                                            int connectTimeout) throws IOException {
        File toFile = null;
        try{
	        toFile = transferToFile(multipartFile);
	        Map<String, Object> data = new HashMap<>();
	        data.put(fileParamName, toFile);
	        data.putAll(otherFormDataMap);
	        long t1 = System.currentTimeMillis();
	        log.info("doPostFormDataFile -- 开始接口调用,otherFormDataMap:{},fileName:{}", JSON.toJSONString(otherFormDataMap), multipartFile.getOriginalFilename());
	        String response = HttpRequest.post(url)
	                .form(data)
	                .contentType("multipart/form-data")
	                .setConnectionTimeout(connectTimeout)
	                .execute()
	                .body();
	        long t2 = System.currentTimeMillis();
	        log.info("doPostFormDataFile -- 调用接口结束,耗时:{},otherFormDataMap:{},fileName:{},响应结果:{}", t2 - t1, JSON.toJSONString(otherFormDataMap), multipartFile.getOriginalFilename(), response);
	        return response;
        }finally{
	        if(toFile != null){
	        	//临时文件使用后,在这里使用delete()删除
	        	toFile.delete();
	        }
        }
    }

    public static File transferToFile(MultipartFile multipartFile) throws IOException {
        //选择用缓冲区来实现这个转换即使用java 创建的临时文件 使用 MultipartFile.transferto()方法 。
        File file = null;
        try {
            String originalFilename = multipartFile.getOriginalFilename();
            //获取文件后缀
            String prefix = originalFilename.substring(originalFilename.lastIndexOf("."));
            //创建临时文件
            file = File.createTempFile(originalFilename, prefix);
            multipartFile.transferTo(file);
            //在这里使用deleteOnExit()会出现文件删除不掉的情况,而使用delete()会导致请求报错
            //file.deleteOnExit();
        } catch (IOException e) {
            log.error("transferToFile--异常:", e);
            throw e;
        }
        return file;
    }
}
;