问题描写叙述
当我们使用SpringMVC作为服务端的框架时,有时不仅仅要应对web前端(jsp、javascript、Jquery等)的訪问请求,有时还可能须要响应Android和JavaSE(桌面应用)这些client的请求。因此,除了web使用form表单或者ajax作为client获取Controller响应之外。纯Java语言向SpringMVC的Controller提供參数和请求结果也是必须要实现的。web前端使用form和ajax来获取Controller响应在本篇博客暂不深究,本篇博课着力于实现纯Javaclient来实现请求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工作原理就是client向 项目地址+Controller前面的value值 这个url发出请求,这个请求会带有一些參数。而SpringMVC会将请求携带的參数转换为Controller相应value以下的方法的參数,然后在这种方法进行相关的操作再返回client一些參数,至于返回的结果就有几种可能了。假设上面没有@ResponseBody凝视,返回的数据有可能是jsp页面的名字,则web会跳转到相应的jsp页面假设是ModelAndView也会跳转到相应的页面而假设加上@ResponseBody。那么返回的就是json类型的数据了,client会接收到Controller返回的參数。那么,明确了springmvc的工作原理,那么怎么使用HttpUrlConnection来訪问Controller就大概清楚了:我们就把请求地址拼接好,然后把携带的数据发送过去就能够等着拿到Controller返回的数据了。
那么第二个问题就来了。请求的地址清楚了。携带的数据应该是什么格式的呢?
刚開始我以为和返回值一样是Json格式的键值对形式,比方说用户登录的userName和userPassword,(Controller的參数形式为 doLogin(String userName,String userPassword))開始我以为应该发送json格式数据。可是服务端不鸟我,没有获得參数。后来我又试了jsonArray形式也不行,试了map的键值对形式不行。我就疑惑了究竟是什么形式的。最后找了半天看了ajax发出相同请求时的http报文结构才知道了数据格式是什么样子的。
是这样的:
中间是一个&连接了两个參数,键值中间是=,好吧,最终明确了。当然这是Controller參数为两个单独的String值的时候的类型,有一种更好的办法是直接把Controller的參数设置为一个,然后在Controller里面解析Json数据,这样事实上更方便一点,以后还更easy做加密工作,只是眼下改起来太麻烦就不改了吧。如今弄明确了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("server返回的结果是:"+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("username或密码错误");
}
};
@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("username密码不能为空");
}
}
});
}
}
好吧。这是非常简陋的代码,毕竟Androidclient才刚刚開始写,activity_login.xml就不放了,两个输入框一个button而已。能够再以下查看输出结果来推断功能是否实现:
以下是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("server返回的数据为:"+ans);
}
}
測试结果为:
要发送的信息是:userName=14121047&userPassword=14121047
server返回的结果是:{"resoult":"success"}
server返回的数据为:{"resoult":"success"}
结束语
当然,我这里演示的知识一个非常简单的android和java向springmvc发送请求的样例,可是,既然中间的道理和方法明确了,那之后不管是什么格式的数据都不是非常难了。
Json数据的那仅仅用一个參数就好了咯,两个參数都传了一个參数不是非常easy么?要传bean的先把Bean转json到服务端再把json转bean就好了咯。233。。
。(这里仅仅是讨论一下两个參数怎么传。并非不想用json。
)
好好学习,天天向上,加油!