当我们使用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 userInfo = new HashMap();
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。)
好好学习,天天向上,加油!