詳解okhttp3 請求頭不能為中文的坑
對源碼有一定的了解,以為基本可以駕馭了,誰知道,坑來了。
問題描述:
上線接口的某一天,有用戶報障,說登錄不上,而且陸續(xù)有報障。
通過log 發(fā)現登錄不上,都有一個類似的報錯:
“java.lang.IllegalArgumentException: Unexpected char 0x514d at 4 in wifiSsid value: "360免費WiFi-DP" at ”
什么?怎么和wifiSsid有關系了?后來查了一下代碼,發(fā)現確實在登錄接口上通過請求頭的形式上傳了 wifissid, 也就是連接的wifi的名字。Exception描述是第四個 char ,也就是“免”字?難道是因為中文的原因?當時是這么猜測的。
源碼查詢
猜測并不能作為判斷的標準,然后做了一下測試,果然連接含義中文名字的wifi就有問題,也就是不能添加中文到請求頭里面。這是為什么呢?最后還是通過源碼進行的求證:
在okhttp的源碼Header.java,發(fā)現set 和add header, 都會有這個判斷:
private void checkNameAndValue(String name, String value) { if (name == null) throw new NullPointerException("name == null"); if (name.isEmpty()) throw new IllegalArgumentException("name is empty"); for (int i = 0, length = name.length(); i < length; i++) { char c = name.charAt(i); if (c <= '\u0020' || c >= '\u007f') { throw new IllegalArgumentException(Util.format( "Unexpected char %#04x at %d in header name: %s", (int) c, i, name)); } } if (value == null) throw new NullPointerException("value == null"); for (int i = 0, length = value.length(); i < length; i++) { char c = value.charAt(i); if ((c <= '\u001f' && c != '\t') || c >= '\u007f') { throw new IllegalArgumentException(Util.format( "Unexpected char %#04x at %d in %s value: %s", (int) c, i, name, value)); } } }
嗷,No,果然是這樣,踩到深坑了。無論是header的 Key 和Value都是不能含有中文的,一旦判定為有中文的出現,就會拋出異常,中斷請求。
后面想了一下,以前使用okhttp2的時候,也有上傳相同的請求頭,為什么就沒有這個問題呢?也同樣去查看了一下okhttp2的源碼,發(fā)現這個判斷是okhttp3才有的,okhttp2其實并沒有這個判斷。
解決方案:
解決方案有以下幾種:
- 不要傳了(當然這個不符合需求)
- 把源碼這個地方改了(也不是太好,不知道會不會引起其他問題)
- 把中文 encode以下(相對較好的辦法):
ssid = URLEncoder.encode(ssid);
總結
遇到問題不可怕,主要是要做好分析,及時解決,多做總結,避免后門再次踩同樣的坑。
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關文章
Android編程自定義搜索框實現方法【附demo源碼下載】
這篇文章主要介紹了Android編程自定義搜索框實現方法,涉及Android界面布局、數據加載、事件響應等相關操作技巧,并附帶完整demo源碼供讀者下載參考,需要的朋友可以參考下2017-12-12Android項目仿UC瀏覽器和360手機衛(wèi)士消息常駐欄(通知欄)
本篇文章主要介紹了Android項目仿UC瀏覽器和360手機衛(wèi)士消息常駐欄(通知欄),可以仿照360的通知欄,有興趣的可以了解一下。2016-11-11