那些年用httpclient時(shí)踩過的一些坑
一、前言
httpclient是java開發(fā)中最常用的工具之一,通常大家會使用其中比較基礎(chǔ)的api去調(diào)用遠(yuǎn)程。長期開發(fā)爬蟲,會接觸httpclient不常用的api,同時(shí)會遇到各式各樣的坑,本文將總結(jié)這些年遇到的坑及相應(yīng)的解決方案。
二、問題及解決方案
問題1:Received fatal alert: handshake_failure
問題背景
開發(fā)某省份移動(dòng)爬蟲時(shí),加載首頁會報(bào)標(biāo)題錯(cuò)誤,嘗試各種辦法都不好使,后來發(fā)現(xiàn)換了jdk1.8就可以了。經(jīng)過長達(dá)一個(gè)星期源碼探尋,發(fā)現(xiàn)錯(cuò)誤源頭是http在握手時(shí),加密算法不支持。
jdk1.8以下版本不支持256位(TLS_DHE_RSA_WITH_AES_256_CBC_SHA )
解決方案
1、下載jce擴(kuò)展包 http://www.oracle.com/technetwork/cn/java/javase/downloads/jce-7-download-432124.html
2、替換/jre/lib/security/里面的兩個(gè)jar
3、覆蓋后如果報(bào)錯(cuò)The jurisdiction policy files are not signed by a trusted signer!,說明下載的版本不對,要下對應(yīng)jdk版本的。
問題2:Certificates does not conformto algorithm constraints
問題背景
用mvn打包時(shí)報(bào)錯(cuò), security.cert.CertificateException: Certificates does not conform toalgorithm constraints
原因是在java1.6之后的這個(gè)配置文件中,認(rèn)為MD2的加密方式安全性太低,因而不支持這種加密方式,同時(shí)也不支持RSA長度小于1024的密文。
需要修改 JAVA_HOME/jre/lib/security/java.security #jdk.certpath.disabledAlgorithms=MD2, RSA keySize < 1024
但是這樣做需要把每臺機(jī)器都改一遍,如果新加機(jī)器忘記改了,就會繼續(xù)報(bào)錯(cuò)。因此需要一套方法,只在代碼層解決問題。
解決方案
經(jīng)查源碼發(fā)現(xiàn)了觸發(fā)問題的代碼位置,通過強(qiáng)制繼承SSLContextBuilder,并強(qiáng)制把private的keymanagers和trustmanagers的值置空就可以解決這個(gè)問題了。
代碼如下:
static class MySSLContextBuilder extends SSLContextBuilder { static final String TLS = "TLS"; static final String SSL = "SSL"; private String protocol; private Set keymanagers; private Set trustmanagers; private SecureRandom secureRandom; public MySSLContextBuilder() { super(); this.keymanagers = new HashSet(); this.trustmanagers = new HashSet(); } }
問題3:超時(shí)時(shí)間不生效
問題背景
很多人在使用httpclient時(shí)會到網(wǎng)上去找例子,例子中經(jīng)常會有類似這樣的設(shè)置
httpGet.getParams().setParameter(ClientPNames.HANDLE_REDIRECTS, !isAutoRelocal);
使用上述方法發(fā)送httpclient,在讀取配置時(shí),如果發(fā)現(xiàn)getParams不為空,則會使得以前設(shè)置的所有參數(shù)都失效,而使用這里設(shè)置的,結(jié)果是導(dǎo)致超時(shí)時(shí)間失效。
解決方案
request.getParams().setParameter
是過期方法,其中每一項(xiàng)參數(shù)在RequestConfig里都有對應(yīng)的,遍歷出來替換一遍即可。
boolean isRedirect = true; if(request != null) { HttpParams params = request.getParams(); if (params instanceof HttpParamsNames) { // 暫時(shí)只支持這個(gè)類型 isRedirect = params.getBooleanParameter( ClientPNames.HANDLE_REDIRECTS, true); } // 清空request request.setParams(new BasicHttpParams()); } if(timeOut > 0) { builder = RequestConfig.custom().setConnectionRequestTimeout(timeOut).setConnectTimeout(timeOut).setSocketTimeout(timeOut).setRedirectsEnabled(isRedirect).setCookieSpec(CookieSpecs.BEST_MATCH); } else { builder = RequestConfig.custom().setConnectionRequestTimeout(connectionTimeout).setConnectTimeout(connectionTimeout).setRedirectsEnabled(isRedirect).setSocketTimeout(socketTimeout).setCookieSpec(CookieSpecs.BEST_MATCH); }
問題4:fildder監(jiān)聽問題
問題背景
開發(fā)爬蟲經(jīng)常會使用fildder來監(jiān)控網(wǎng)絡(luò)請求,但是使用httpclient時(shí)想用fildder會很難,網(wǎng)上查各種辦法都不好用。
下面為大家來排個(gè)錯(cuò),使用下面方法就可以完美解決這個(gè)問題,讓fildder監(jiān)控更容易。
解決方案
首先java端
// client builder HttpClientBuilder builder = HttpClients.custom(); if(useFidder) { // 默認(rèn)fidder寫死 builder.setProxy(new HttpHost("127.0.0.1", 8888)); }
fildder端
tools->fiddler options->https->actions->export root certificate to ... \bin\keytool.exe -import -file C:\Users\\Desktop\FiddlerRoot.cer -keystore FiddlerKeystore -alias Fiddler
問題5:支持gzip
問題及解決方案
有些網(wǎng)站返回進(jìn)行了gzip壓縮,返回內(nèi)容是壓縮的結(jié)果,需要解壓。
代碼如下:
HttpClient wrappedHttpClient = builder.setUserAgent(requestUA) .addInterceptorLast(new HttpResponseInterceptor() { @Override public void process(HttpResponse httpResponse, HttpContext httpContext) throws HttpException, IOException { HttpEntity httpEntity = httpResponse.getEntity(); Header header = httpEntity.getContentEncoding(); if (header != null) { for (HeaderElement element : header.getElements()) { if ("gzip".equalsIgnoreCase(element.getName())) { httpResponse.setEntity(new GzipDecompressingEntity(httpResponse.getEntity())); } } } } })
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對腳本之家的支持。
- java使用httpclient模擬post請求和get請求示例
- java使用httpclient發(fā)送post請求示例
- 使用httpclient無需證書調(diào)用https的示例(java調(diào)用https)
- JAVA利用HttpClient進(jìn)行POST請求(HTTPS)實(shí)例
- java發(fā)送HttpClient請求及接收請求結(jié)果過程的簡單實(shí)例
- java實(shí)現(xiàn)HttpClient異步請求資源的方法
- 使用java的HttpClient實(shí)現(xiàn)多線程并發(fā)
- HttpClient 在Java項(xiàng)目中的使用詳解
- Java利用HttpClient模擬POST表單操作應(yīng)用及注意事項(xiàng)
- java web中 HttpClient模擬瀏覽器登錄后發(fā)起請求
相關(guān)文章
spring-data-jpa實(shí)現(xiàn)增刪改查以及分頁操作方法
下面小編就為大家分享一篇spring-data-jpa實(shí)現(xiàn)增刪改查以及分頁操作方法,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-02-02Spring Security驗(yàn)證流程剖析及自定義驗(yàn)證方法
Spring Security是一個(gè)能夠?yàn)榛赟pring的企業(yè)應(yīng)用系統(tǒng)提供聲明式的安全訪問控制解決方案的安全框架。這篇文章主要介紹了Spring Security驗(yàn)證流程剖析及自定義驗(yàn)證方法,需要的朋友可以參考下2018-03-03Spring MVC深入學(xué)習(xí)之啟動(dòng)初始化過程
最近因?yàn)楣ぷ鞯脑蛟趯W(xué)習(xí)Spring MVC,為了更深入的學(xué)習(xí)Spring MVC,下面這篇文章主要給大家介紹了關(guān)于Spring MVC深入學(xué)習(xí)之啟動(dòng)初始化過程的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起看看吧。2017-07-07java遍歷http請求request的所有參數(shù)實(shí)現(xiàn)方法
下面小編就為大家?guī)硪黄猨ava遍歷http請求request的所有參數(shù)實(shí)現(xiàn)方法。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2016-09-09SpringBoot視圖解析實(shí)現(xiàn)原理深入分析
視圖解析其實(shí)就是SpringBoot某一個(gè)controller的方法執(zhí)行完成之后,它是跳轉(zhuǎn)到那個(gè)頁面。由于我們springboot項(xiàng)目默認(rèn)打包為jar包,是形成壓縮包的形式,而jsp又不支持壓縮,所以我們SpringBoot不知JSP的,需要引入第三方模板引擎才可以處理2022-10-10使用Get方式提交數(shù)據(jù)到Tomcat服務(wù)器的方法
這篇文章將介紹向服務(wù)器發(fā)送數(shù)據(jù),并且服務(wù)器將數(shù)據(jù)的處理結(jié)果返回給客戶端,本文給大家介紹使用Get方式向服務(wù)器發(fā)送數(shù)據(jù),感興趣的朋友一起學(xué)習(xí)吧2016-04-04