Android(或者Java)通過HttpUrlConnection向SpringMVC請求資料(資料繫結)
問題描述
當我們使用SpringMVC作為服務端的框架時,有時不僅僅要應對web前端(jsp、javascript、Jquery等)的訪問請求,有時還可能需要響應Android和JavaSE(桌面應用)這些客戶端的請求,因此,除了web使用form表單或者ajax作為客戶端獲取Controller響應之外,純Java語言向SpringMVC的Controller提供引數和請求結果也是必須要實現的。web前端使用form和ajax來獲取Controller響應在本篇部落格暫不深究,本篇博課著力於實現純Java客戶端來實現請求Controller並獲取響應。(如果不想看下面的方法探索,可以直接跳到第三部分)
方法探索
確定了需要解決的問題之後,我開始搜尋相關的部落格和問答,發現這種問題相關解決方法不是很多,找了很多部落格都是牛頭不對馬嘴,但是還是有一些對解決問題有幫助的內容,之後又翻了很久的書,對這個問題的解決方法有了基本的認識。
看網上說的行之有效的方法是使用HttpClient來實現這個功能,不過需要匯入HttpClient和HttpCore兩個jar包,當時研究了一下覺得可行。但是在實際操作過程中,在向Android(IDE:Android Studio)build.gradle裡面匯入上述兩個jar包的依賴後,同步的時候Android Studio給出了兩個警告,大意是HttpClient已經過期云云,然後我又專門去查了一個HttpClient這個東西和Android之間的關係,發現Android6.0以及之後的版本放棄了對HttpClient的支援,強行匯入jar包也不行。(據說有人用了非常麻煩的方法匯入成功過 = =)那好嘞,不讓用算了,那就不用唄。看參考書《瘋狂Android講義》對HttpClient還那麼推崇,也是有點醉。具體不讓用的原因就不再深究了,接下來就研究使用HttpUrlConnection來解決上述問題吧。
在使用HttpURLConnection的過程中程式碼實現照著套路來問題不大,遇到的問題是SpringMVC的Controller怎麼接收資料,接收的資料格式是什麼?我應該傳什麼型別的資料?
Controller怎麼接收資料這個問題經過前段時間的學習我已經有個大概的認識,SpringMVC的Controller工作原理就是客戶端向 專案地址+Controller前面的value值 這個url發出請求,這個請求會帶有一些引數,而SpringMVC會將請求攜帶的引數轉換為Controller對應value下面的方法的引數,然後在這個方法進行相關的操作再返回客戶端一些引數,至於返回的結果就有幾種可能了。如果上面沒有@ResponseBody註釋,返回的資料有可能是jsp頁面的名字,則web會跳轉到對應的jsp頁面如果是ModelAndView也會跳轉到對應的頁面而如果加上@ResponseBody,那麼返回的就是json型別的資料了,客戶端會接收到Controller返回的引數。那麼,明白了springmvc的工作原理,那麼怎麼使用HttpUrlConnection來訪問Controller就大概清楚了:我們就把請求地址拼接好,然後把攜帶的資料傳送過去就可以等著拿到Controller返回的資料了。那麼第二個問題就來了,請求的地址清楚了,攜帶的資料應該是什麼格式的呢?
剛開始我以為和返回值一樣是Json格式的鍵值對形式,比如說使用者登入的userName和userPassword,(Controller的引數形式為 doLogin(String userName,String userPassword))開始我以為應該傳送json格式資料,可是服務端不鳥我,沒有獲得引數。後來我又試了jsonArray形式也不行,試了map的鍵值對形式不行。我就疑惑了到底是什麼形式的。最後找了半天看了ajax發出同樣請求時的http報文結構才知道了資料格式是什麼樣子的。是這樣的:
中間是一個&連線了兩個引數,鍵值中間是=,好吧,終於明白了。當然這是Controller引數為兩個單獨的String值的時候的型別,有一種更好的辦法是直接把Controller的引數設定為一個,然後在Controller裡面解析Json資料,這樣其實更方便一點,以後還更容易做加密工作,不過目前改起來太麻煩就不改了吧,現在弄明白了controller是兩個引數的時候是這種結構的。至於其它的list,bean以及json型別這裡就不探究了,因為我看到很多別的博主都講了怎麼轉了。廢話了這麼多,下面放程式碼。
解決問題
SpringMVC對應Controller程式碼(result單詞拼錯了,懶得改咯):
@RequestMapping(value="/doLogin",method=RequestMethod.POST)
@ResponseBody
public Map<String,Object> doLogin(String userName,String userPassword,HttpSession httpSession){
String resoult="fail";
User user = userService.getUser(userName);
UserDetail userDetail = userDetailService.getUserDetail(userName);
if(user!=null){
if(Objects.equals(userDetail.getUserDetailPassword(), userPassword)){
httpSession.setAttribute("currentUser",user);
resoult = "success";
}
else{
resoult = "wrong";
}
}
else{
resoult = "unexist";
}
Map<String, Object> resoults = new HashMap<String,Object>();
resoults.put("resoult", resoult);
return resoults;
}
Android端或者java端寫的一個HttpPostUtils工具類
package cn.justwithme.withme.Utils;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
import java.io.PrintWriter;
import java.net.HttpURLConnection;
import java.net.MalformedURLException;
import java.net.URL;
/**
* Created by 桐木木 on 2017/2/8.
* 傳送Http請求,獲取Controller返回的資料
*
*/
public class HttpPostUtils {
public static String doPostRequest(String path,Object content){
PrintWriter out = null;
BufferedReader in = null;
String resoult = "";
try {
System.out.println("要傳送的資訊是:"+content);
/*拼接url,Android這裡需要換上遠端地址,因為Android端沒辦法訪問localhost,java的話本地tomcat執行的話倒是無妨*/
String address = http://localhost:8080/WithMe/"+path;
URL url = new URL(address);
HttpURLConnection httpURLConnection = (HttpURLConnection)url.openConnection();
//這兩個引數必須加
httpURLConnection.setDoInput(true);
httpURLConnection.setDoOutput(true);
//設定超時時間
httpURLConnection.setReadTimeout(10*1000);
httpURLConnection.setConnectTimeout(10*1000);
httpURLConnection.setRequestMethod("POST");
httpURLConnection.connect();
out = new PrintWriter(httpURLConnection.getOutputStream());
//在輸出流中寫入引數
out.print(content);
out.flush();
if(httpURLConnection.getResponseCode() == 200){
in = new BufferedReader(new InputStreamReader(httpURLConnection.getInputStream(),"utf-8"));
String line;
while((line=in.readLine())!=null){
resoult+=line;
}
}
System.out.println("伺服器返回的結果是:"+resoult);
return resoult;
} catch (MalformedURLException e) {
System.out.println("URL異常");
e.printStackTrace();
} catch (IOException e) {
System.out.println("IO異常");
e.printStackTrace();
}finally {
try{
if(out!=null)
out.close();
if(in!=null)
in.close();
}catch (IOException e) {
e.printStackTrace();
}
}
return null;
}
}
工具類寫好之後我們分別在Android端和Java端進行一些測試,(在我的服務端我已經註冊過14121047的前提下,你如果只是測試把服務端查詢資料庫校驗密碼改成字串匹配就成了)。
Android
LoginActivity.java
package cn.justwithme.withme.Activity;
import android.os.Bundle;
import android.os.Handler;
import android.os.Message;
import android.support.v7.app.AppCompatActivity;
import android.view.View;
import android.widget.Button;
import android.widget.EditText;
import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.JSONObject;
import java.util.HashMap;
import java.util.Map;
import cn.justwithme.withme.R;
import cn.justwithme.withme.Utils.HttpPostUtils;
public class LoginActivity extends AppCompatActivity {
private String result;
private Handler mHanlder = new Handler() {
@Override
public void handleMessage(Message message) {
result = (String) message.obj;
JSONObject resultJson = JSON.parseObject(result);
String finalResult = resultJson.getString("resoult");
System.out.println("結果是:"+finalResult);
if(finalResult.equals("success")){
System.out.println("登陸成功");
}
else
System.out.println("使用者名稱或密碼錯誤");
}
};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_login);
final EditText userNameInput = (EditText)findViewById(R.id.userName);
final EditText passwordInput = (EditText)findViewById(R.id.password);
Button login = (Button)findViewById(R.id.login);
login.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
String userName = userNameInput.getText().toString();
String userPassword = passwordInput.getText().toString();
final Map<String,Object> userInfo = new HashMap<String,Object>();
userInfo.put("userName",userName);
userInfo.put("userPassword",userPassword);
// UserLogin userInfo = new UserLogin();
// userInfo.setUserName(userName);
// userInfo.setUserPassword(userPassword);
final String user = "userName="+userName+"&userPassword="+userPassword;
if(userName!= null && !userName.equals("") && userPassword!=null && !userPassword.equals("")){
/*這裡需要留意的是httpPostUtils請求在Android裡面不能放在主執行緒裡面,必須新建一個子執行緒,然後通過Hanlder把子執行緒的值傳過來更新UI(因為子執行緒不能直接更改UI)*/
new Thread() {
public void run() {
String response = HttpPostUtils.doPostRequest("doLogin",user);
Message message = Message.obtain();
message.obj = response;
mHanlder.sendMessage(message);
}
}.start();
}
else{
System.out.println("使用者名稱密碼不能為空");
}
}
});
}
}
好吧,這是非常簡陋的程式碼,畢竟Android客戶端才剛剛開始寫,activity_login.xml就不放了,兩個輸入框一個按鈕而已,可以再下面檢視輸出結果來判斷功能是否實現:
下面是Java的測試程式碼,用的工具類和Android用的一樣的,
package httpPost;
public class test {
public static void main(String[] args){
String test = "userName=14121047&userPassword=14121047";
String ans = HttpPostUtils.doPostRequest("doLogin", test);
System.out.println("伺服器返回的資料為:"+ans);
}
}
測試結果為:
要傳送的資訊是:userName=14121047&userPassword=14121047
伺服器返回的結果是:{"resoult":"success"}
伺服器返回的資料為:{"resoult":"success"}
結束語
當然,我這裡演示的知識一個非常簡單的android和java向springmvc傳送請求的例子,但是,既然中間的道理和方法明白了,那之後無論是什麼格式的資料都不是很難了。Json資料的那隻用一個引數就好了咯,兩個引數都傳了一個引數不是很容易麼?要傳bean的先把Bean轉json到服務端再把json轉bean就好了咯。233。。。(這裡只是討論一下兩個引數怎麼傳,並不是不想用json。)
好好學習,天天向上,加油!