在分布式環(huán)境中,如何支持PC朽缎、APP(ios惨远、android)等多端的會(huì)話共享,這也是所有公司都需要的解決方案话肖,用傳統(tǒng)的session方式來(lái)解決北秽,我想已經(jīng)out了,我們是否可以找一個(gè)通用的方案狼牺,比如用傳統(tǒng)cas來(lái)實(shí)現(xiàn)多系統(tǒng)之間的sso單點(diǎn)登錄或使用oauth的第三方登錄方案? 今天給大家簡(jiǎn)單講解一下使用spring攔截器Interceptor機(jī)制羡儿、jwt認(rèn)證方式、redis分布式緩存實(shí)現(xiàn)sso單點(diǎn)登錄
}
publicvoid?setUid(String?uid)?{
this.uid?=?uid;
}
public?String?getToken()?{
return?token;
}
publicvoid?setToken(String?token)?{
this.token?=?token;
}
publiclong?getRefTime()?{
return?refTime;
}
publicvoid?setRefTime(long?refTime)?{
this.refTime?=?refTime;
}
}
public class RedisLogin implements Serializable{
/**
*
*/
private static final long serialVersionUID = 8116817810829835862L;
/**
* 用戶id
*/
private String uid;
/**
* jwt生成的token信息
*/
private String token;
/**
* 登錄或刷新應(yīng)用的時(shí)間
*/
private long refTime;
public RedisLogin(){
}
public RedisLogin(String uid, String token, long refTime){
this.uid = uid;
this.token = token;
this.refTime = refTime;
}
public String getUid() {
return uid;
}
public void setUid(String uid) {
this.uid = uid;
}
public String getToken() {
return token;
}
public void setToken(String token) {
this.token = token;
}
public long getRefTime() {
return refTime;
}
public void setRefTime(long refTime) {
this.refTime = refTime;
}
}
6. 編寫(xiě)LoginInterceptor.java攔截器
Java代碼
publicclass?LoginInterceptorimplements?HandlerInterceptor{
publicboolean?preHandle(HttpServletRequest?request,?HttpServletResponse?response,?Object?handler)
throws?Exception?{
PrintWriter?writer?=null;
HandlerMethod?method?=null;
try?{
method?=?(HandlerMethod)?handler;
}catch?(Exception?e)?{
writer?=?response.getWriter();
ResponseVO?responseVO?=?ResponseCode.buildEnumResponseVO(ResponseCode.REQUEST_URL_NOT_SERVICE,false);
responseMessage(response,?writer,?responseVO);
returnfalse;
}
IsLogin?isLogin?=?method.getMethodAnnotation(IsLogin.class);
if(null?==?isLogin){
returntrue;
}
response.setCharacterEncoding("utf-8");
String?token?=?request.getHeader("token");
String?uid?=?request.getHeader("uid");
//token不存在
if(StringUtils.isEmpty(token))?{
writer?=?response.getWriter();
ResponseVO?responseVO?=?LoginResponseCode.buildEnumResponseVO(LoginResponseCode.LOGIN_TOKEN_NOT_NULL,false);
responseMessage(response,?writer,?responseVO);
returnfalse;
}
if(StringUtils.isEmpty(uid)){
writer?=?response.getWriter();
ResponseVO?responseVO?=?LoginResponseCode.buildEnumResponseVO(LoginResponseCode.USERID_NOT_NULL,false);
responseMessage(response,?writer,?responseVO);
returnfalse;
}
Login?login?=?JWT.unsign(token,?Login.class);
//解密token后的loginId與用戶傳來(lái)的loginId判斷是否一致
if(null?==?login?||?!StringUtils.equals(login.getUid(),?uid)){
writer?=?response.getWriter();
ResponseVO?responseVO?=?LoginResponseCode.buildEnumResponseVO(LoginResponseCode.USERID_NOT_UNAUTHORIZED,false);
responseMessage(response,?writer,?responseVO);
returnfalse;
}
//驗(yàn)證登錄時(shí)間
RedisLogin?redisLogin?=?(RedisLogin)JedisUtils.getObject(uid);
if(null?==?redisLogin){
writer?=?response.getWriter();
ResponseVO?responseVO?=?LoginResponseCode.buildEnumResponseVO(LoginResponseCode.RESPONSE_CODE_UNLOGIN_ERROR,false);
responseMessage(response,?writer,?responseVO);
returnfalse;
}
if(!StringUtils.equals(token,?redisLogin.getToken())){
writer?=?response.getWriter();
ResponseVO?responseVO?=?LoginResponseCode.buildEnumResponseVO(LoginResponseCode.USERID_NOT_UNAUTHORIZED,false);
responseMessage(response,?writer,?responseVO);
returnfalse;
}
//系統(tǒng)時(shí)間>有效期(說(shuō)明已經(jīng)超過(guò)有效期)
if?(System.currentTimeMillis()?>?redisLogin.getRefTime())?{
writer?=?response.getWriter();
ResponseVO?responseVO?=?LoginResponseCode.buildEnumResponseVO(LoginResponseCode.LOGIN_TIME_EXP,false);
responseMessage(response,?writer,?responseVO);
returnfalse;
}
//重新刷新有效期
redisLogin?=new?RedisLogin(uid,?token,?System.currentTimeMillis()?+?60L*?1000L*?30L);
JedisUtils.setObject(uid?,?redisLogin,360000000);
returntrue;
}
publicvoid?postHandle(HttpServletRequest?request,?HttpServletResponse?response,?Object?handler,
ModelAndView?modelAndView)throws?Exception?{
}
publicvoid?afterCompletion(HttpServletRequest?request,?HttpServletResponse?response,?Object?handler,?Exception?ex)
throws?Exception?{
}
privatevoid?responseMessage(HttpServletResponse?response,?PrintWriter?out,?ResponseVO?responseVO)?{
response.setContentType("application/json;?charset=utf-8");
JSONObject?result?=new?JSONObject();
result.put("result",?responseVO);
out.print(result);
out.flush();
out.close();
}
}
[java]view plaincopyprint?
publicclass?LoginInterceptorimplements?HandlerInterceptor{
publicboolean?preHandle(HttpServletRequest?request,?HttpServletResponse?response,?Object?handler)
throws?Exception?{
PrintWriter?writer?=null;
HandlerMethod?method?=null;
try?{
method?=?(HandlerMethod)?handler;
}catch?(Exception?e)?{
writer?=?response.getWriter();
ResponseVO?responseVO?=?ResponseCode.buildEnumResponseVO(ResponseCode.REQUEST_URL_NOT_SERVICE,false);
responseMessage(response,?writer,?responseVO);
returnfalse;
}
IsLogin?isLogin?=?method.getMethodAnnotation(IsLogin.class);
if(null?==?isLogin){
returntrue;
}
response.setCharacterEncoding("utf-8");
String?token?=?request.getHeader("token");
String?uid?=?request.getHeader("uid");
//token不存在
if(StringUtils.isEmpty(token))?{
writer?=?response.getWriter();
ResponseVO?responseVO?=?LoginResponseCode.buildEnumResponseVO(LoginResponseCode.LOGIN_TOKEN_NOT_NULL,false);
responseMessage(response,?writer,?responseVO);
returnfalse;
}
if(StringUtils.isEmpty(uid)){
writer?=?response.getWriter();
ResponseVO?responseVO?=?LoginResponseCode.buildEnumResponseVO(LoginResponseCode.USERID_NOT_NULL,false);
responseMessage(response,?writer,?responseVO);
returnfalse;
}
Login?login?=?JWT.unsign(token,?Login.class);
//解密token后的loginId與用戶傳來(lái)的loginId判斷是否一致
if(null?==?login?||?!StringUtils.equals(login.getUid(),?uid)){
writer?=?response.getWriter();
ResponseVO?responseVO?=?LoginResponseCode.buildEnumResponseVO(LoginResponseCode.USERID_NOT_UNAUTHORIZED,false);
responseMessage(response,?writer,?responseVO);
returnfalse;
}
//驗(yàn)證登錄時(shí)間
RedisLogin?redisLogin?=?(RedisLogin)JedisUtils.getObject(uid);
if(null?==?redisLogin){
writer?=?response.getWriter();
ResponseVO?responseVO?=?LoginResponseCode.buildEnumResponseVO(LoginResponseCode.RESPONSE_CODE_UNLOGIN_ERROR,false);
responseMessage(response,?writer,?responseVO);
returnfalse;
}
if(!StringUtils.equals(token,?redisLogin.getToken())){
writer?=?response.getWriter();
ResponseVO?responseVO?=?LoginResponseCode.buildEnumResponseVO(LoginResponseCode.USERID_NOT_UNAUTHORIZED,false);
responseMessage(response,?writer,?responseVO);
returnfalse;
}
//系統(tǒng)時(shí)間>有效期(說(shuō)明已經(jīng)超過(guò)有效期)
if?(System.currentTimeMillis()?>?redisLogin.getRefTime())?{
writer?=?response.getWriter();
ResponseVO?responseVO?=?LoginResponseCode.buildEnumResponseVO(LoginResponseCode.LOGIN_TIME_EXP,false);
responseMessage(response,?writer,?responseVO);
returnfalse;
}
//重新刷新有效期
redisLogin?=new?RedisLogin(uid,?token,?System.currentTimeMillis()?+?60L*?1000L*?30L);
JedisUtils.setObject(uid?,?redisLogin,360000000);
returntrue;
}
publicvoid?postHandle(HttpServletRequest?request,?HttpServletResponse?response,?Object?handler,
ModelAndView?modelAndView)throws?Exception?{
}
publicvoid?afterCompletion(HttpServletRequest?request,?HttpServletResponse?response,?Object?handler,?Exception?ex)
throws?Exception?{
}
privatevoid?responseMessage(HttpServletResponse?response,?PrintWriter?out,?ResponseVO?responseVO)?{
response.setContentType("application/json;?charset=utf-8");
JSONObject?result?=new?JSONObject();
result.put("result",?responseVO);
out.print(result);
out.flush();
out.close();
}
}
public class LoginInterceptor implements HandlerInterceptor{
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)
throws Exception {
PrintWriter writer = null;
HandlerMethod method = null;
try {
method = (HandlerMethod) handler;
} catch (Exception e) {
writer = response.getWriter();
ResponseVO responseVO = ResponseCode.buildEnumResponseVO(ResponseCode.REQUEST_URL_NOT_SERVICE, false);
responseMessage(response, writer, responseVO);
return false;
}
IsLogin isLogin = method.getMethodAnnotation(IsLogin.class);
if(null == isLogin){
return true;
}
response.setCharacterEncoding("utf-8");
String token = request.getHeader("token");
String uid = request.getHeader("uid");
//token不存在
if(StringUtils.isEmpty(token)) {
writer = response.getWriter();
ResponseVO responseVO = LoginResponseCode.buildEnumResponseVO(LoginResponseCode.LOGIN_TOKEN_NOT_NULL, false);
responseMessage(response, writer, responseVO);
return false;
}
if(StringUtils.isEmpty(uid)){
writer = response.getWriter();
ResponseVO responseVO = LoginResponseCode.buildEnumResponseVO(LoginResponseCode.USERID_NOT_NULL, false);
responseMessage(response, writer, responseVO);
return false;
}
Login login = JWT.unsign(token, Login.class);
//解密token后的loginId與用戶傳來(lái)的loginId判斷是否一致
if(null == login || !StringUtils.equals(login.getUid(), uid)){
writer = response.getWriter();
ResponseVO responseVO = LoginResponseCode.buildEnumResponseVO(LoginResponseCode.USERID_NOT_UNAUTHORIZED, false);
responseMessage(response, writer, responseVO);
return false;
}
//驗(yàn)證登錄時(shí)間
RedisLogin redisLogin = (RedisLogin)JedisUtils.getObject(uid);
if(null == redisLogin){
writer = response.getWriter();
ResponseVO responseVO = LoginResponseCode.buildEnumResponseVO(LoginResponseCode.RESPONSE_CODE_UNLOGIN_ERROR, false);
responseMessage(response, writer, responseVO);
return false;
}
if(!StringUtils.equals(token, redisLogin.getToken())){
writer = response.getWriter();
ResponseVO responseVO = LoginResponseCode.buildEnumResponseVO(LoginResponseCode.USERID_NOT_UNAUTHORIZED, false);
responseMessage(response, writer, responseVO);
return false;
}
//系統(tǒng)時(shí)間>有效期(說(shuō)明已經(jīng)超過(guò)有效期)
if (System.currentTimeMillis() > redisLogin.getRefTime()) {
writer = response.getWriter();
ResponseVO responseVO = LoginResponseCode.buildEnumResponseVO(LoginResponseCode.LOGIN_TIME_EXP, false);
responseMessage(response, writer, responseVO);
return false;
}
//重新刷新有效期
redisLogin = new RedisLogin(uid, token, System.currentTimeMillis() + 60L* 1000L* 30L);
JedisUtils.setObject(uid , redisLogin, 360000000);
return true;
}
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler,
ModelAndView modelAndView) throws Exception {
}
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)
throws Exception {
}
private void responseMessage(HttpServletResponse response, PrintWriter out, ResponseVO responseVO) {
response.setContentType("application/json; charset=utf-8");
JSONObject result = new JSONObject();
result.put("result", responseVO);
out.print(result);
out.flush();
out.close();
}
}
7. 定義異常的LoginResponseCode
Java代碼
publicenum?LoginResponseCode?{
USERID_NOT_NULL(3001,"用戶id不能為空."),
LOGIN_TOKEN_NOT_NULL(3002,"登錄token不能為空."),
USERID_NOT_UNAUTHORIZED(3003,"用戶token或ID驗(yàn)證不通過(guò)"),
RESPONSE_CODE_UNLOGIN_ERROR(421,"未登錄異常"),
LOGIN_TIME_EXP(3004,"登錄時(shí)間超長(zhǎng)是钥,請(qǐng)重新登錄");
//?成員變量
privateint?code;//狀態(tài)碼
private?String?message;//返回消息
//?構(gòu)造方法
private?LoginResponseCode(int?code,String?message)?{
this.code?=?code;
this.message?=?message;
}
publicint?getCode()?{
return?code;
}
publicvoid?setCode(int?code)?{
this.code?=?code;
}
public?String?getMessage()?{
return?message;
}
publicvoid?setMessage(String?message)?{
this.message?=?message;
}
publicstatic?ResponseVO?buildEnumResponseVO(LoginResponseCode?responseCode,?Object?data)?{
returnnew?ResponseVO(responseCode.getCode(),responseCode.getMessage(),data);
}
publicstatic?Map?buildReturnMap(LoginResponseCode?responseCode,?Object?data)?{
Map?map?=new?HashMap();
map.put("code",?responseCode.getCode());
map.put("message",?responseCode.getMessage());
map.put("data",?data);
return?map;
}
}
[java]view plaincopyprint?
publicenum?LoginResponseCode?{
USERID_NOT_NULL(3001,"用戶id不能為空."),
LOGIN_TOKEN_NOT_NULL(3002,"登錄token不能為空."),
USERID_NOT_UNAUTHORIZED(3003,"用戶token或ID驗(yàn)證不通過(guò)"),
RESPONSE_CODE_UNLOGIN_ERROR(421,"未登錄異常"),
LOGIN_TIME_EXP(3004,"登錄時(shí)間超長(zhǎng)掠归,請(qǐng)重新登錄");
//?成員變量
privateint?code;//狀態(tài)碼
private?String?message;//返回消息
//?構(gòu)造方法
private?LoginResponseCode(int?code,String?message)?{
this.code?=?code;
this.message?=?message;
}
publicint?getCode()?{
return?code;
}
publicvoid?setCode(int?code)?{
this.code?=?code;
}
public?String?getMessage()?{
return?message;
}
publicvoid?setMessage(String?message)?{
this.message?=?message;
}
publicstatic?ResponseVO?buildEnumResponseVO(LoginResponseCode?responseCode,?Object?data)?{
returnnew?ResponseVO(responseCode.getCode(),responseCode.getMessage(),data);
}
publicstatic?Map?buildReturnMap(LoginResponseCode?responseCode,?Object?data)?{
Map?map?=new?HashMap();
map.put("code",?responseCode.getCode());
map.put("message",?responseCode.getMessage());
map.put("data",?data);
return?map;
}
}
public enum LoginResponseCode {
USERID_NOT_NULL(3001,"用戶id不能為空."),
LOGIN_TOKEN_NOT_NULL(3002,"登錄token不能為空."),
USERID_NOT_UNAUTHORIZED(3003, "用戶token或ID驗(yàn)證不通過(guò)"),
RESPONSE_CODE_UNLOGIN_ERROR(421, "未登錄異常"),
LOGIN_TIME_EXP(3004, "登錄時(shí)間超長(zhǎng)缅叠,請(qǐng)重新登錄");
// 成員變量
private int code; //狀態(tài)碼
private String message; //返回消息
// 構(gòu)造方法
private LoginResponseCode(int code,String message) {
this.code = code;
this.message = message;
}
public int getCode() {
return code;
}
public void setCode(int code) {
this.code = code;
}
public String getMessage() {
return message;
}
public void setMessage(String message) {
this.message = message;
}
public static ResponseVO buildEnumResponseVO(LoginResponseCode responseCode, Object data) {
return new ResponseVO(responseCode.getCode(),responseCode.getMessage(),data);
}
public static Map buildReturnMap(LoginResponseCode responseCode, Object data) {
Map map = new HashMap();
map.put("code", responseCode.getCode());
map.put("message", responseCode.getMessage());
map.put("data", data);
return map;
}
}
8. 編寫(xiě)統(tǒng)一sso單點(diǎn)登錄接口:
Java代碼
@RequestMapping(value?="/login",?method?=?RequestMethod.POST)
public?Map?login(@RequestBody?JSONObject?json){
String?loginName?=?json.optString("loginName");
String?password?=?json.optString("password");
//校驗(yàn)用戶名不能為空
if(StringUtils.isEmpty(loginName)){
return?MemberResponseCode.buildReturnMap(MemberResponseCode.RESPONSE_CODE_USER_NAME_IS_NOT_EMPTY,null);
}
//校驗(yàn)用戶密碼不能為空
if(StringUtils.isEmpty(password)){
return?MemberResponseCode.buildReturnMap(MemberResponseCode.RESPONSE_CODE_PWD_CAN_NOT_BE_EMPTY,null);
}
//根據(jù)用戶名查詢數(shù)據(jù)庫(kù)用戶信息
User?user?=?systemService.getBaseUserByLoginName(loginName);
//用戶名或密碼不正確
if(null?==?user){
return?MemberResponseCode.buildReturnMap(MemberResponseCode.RESPONSE_CODE_USER_VALIDATE_NO_SUCCESS,false);
}
boolean?isValidate?=?systemService.validatePassword(password,?user.getPassword());
if(!isValidate){
return?MemberResponseCode.buildReturnMap(MemberResponseCode.RESPONSE_CODE_USER_VALIDATE_NO_SUCCESS,false);
}
if(isValidate){
//HttpSession?session?=request.getSession(false);
Login?login?=new?Login(user.getId(),?user.getLoginName(),?user.getPassword());
//給用戶jwt加密生成token
String?token?=?JWT.sign(login,?60L*?1000L*?30L);
Map?result?=new?HashMap();
result.put("loginToken",?token);
result.put("userId",?user.getId());
result.put("user",?user);
//保存用戶信息到session
//session.setAttribute(user.getId()?+?"@@"?+?token,?user);
//重建用戶信息
this.rebuildLoginUser(user.getId(),?token);
return?ResponseCode.buildReturnMap(ResponseCode.RESPONSE_CODE_LOGIN_SUCCESS,?result);
}
return?ResponseCode.buildReturnMap(ResponseCode.USER_LOGIN_PWD_ERROR,false);
}
[java]view plaincopyprint?
@RequestMapping(value?="/login",?method?=?RequestMethod.POST)
public?Map?login(@RequestBody?JSONObject?json){
String?loginName?=?json.optString("loginName");
String?password?=?json.optString("password");
//校驗(yàn)用戶名不能為空
if(StringUtils.isEmpty(loginName)){
return?MemberResponseCode.buildReturnMap(MemberResponseCode.RESPONSE_CODE_USER_NAME_IS_NOT_EMPTY,null);
}
//校驗(yàn)用戶密碼不能為空
if(StringUtils.isEmpty(password)){
return?MemberResponseCode.buildReturnMap(MemberResponseCode.RESPONSE_CODE_PWD_CAN_NOT_BE_EMPTY,null);
}
//根據(jù)用戶名查詢數(shù)據(jù)庫(kù)用戶信息
User?user?=?systemService.getBaseUserByLoginName(loginName);
//用戶名或密碼不正確
if(null?==?user){
return?MemberResponseCode.buildReturnMap(MemberResponseCode.RESPONSE_CODE_USER_VALIDATE_NO_SUCCESS,false);
}
boolean?isValidate?=?systemService.validatePassword(password,?user.getPassword());
if(!isValidate){
return?MemberResponseCode.buildReturnMap(MemberResponseCode.RESPONSE_CODE_USER_VALIDATE_NO_SUCCESS,false);
}
if(isValidate){
//HttpSession?session?=request.getSession(false);
Login?login?=new?Login(user.getId(),?user.getLoginName(),?user.getPassword());
//給用戶jwt加密生成token
String?token?=?JWT.sign(login,?60L*?1000L*?30L);
Map?result?=new?HashMap();
result.put("loginToken",?token);
result.put("userId",?user.getId());
result.put("user",?user);
//保存用戶信息到session
//session.setAttribute(user.getId()?+?"@@"?+?token,?user);
//重建用戶信息
this.rebuildLoginUser(user.getId(),?token);
return?ResponseCode.buildReturnMap(ResponseCode.RESPONSE_CODE_LOGIN_SUCCESS,?result);
}
return?ResponseCode.buildReturnMap(ResponseCode.USER_LOGIN_PWD_ERROR,false);
}
@RequestMapping(value = "/login", method = RequestMethod.POST)
public Map login(@RequestBody JSONObject json){
String loginName = json.optString("loginName");
String password = json.optString("password");
//校驗(yàn)用戶名不能為空
if(StringUtils.isEmpty(loginName)){
return MemberResponseCode.buildReturnMap(MemberResponseCode.RESPONSE_CODE_USER_NAME_IS_NOT_EMPTY, null);
}
//校驗(yàn)用戶密碼不能為空
if(StringUtils.isEmpty(password)){
return MemberResponseCode.buildReturnMap(MemberResponseCode.RESPONSE_CODE_PWD_CAN_NOT_BE_EMPTY, null);
}
//根據(jù)用戶名查詢數(shù)據(jù)庫(kù)用戶信息
User user = systemService.getBaseUserByLoginName(loginName);
//用戶名或密碼不正確
if(null == user){
return MemberResponseCode.buildReturnMap(MemberResponseCode.RESPONSE_CODE_USER_VALIDATE_NO_SUCCESS, false);
}
boolean isValidate = systemService.validatePassword(password, user.getPassword());
if(!isValidate){
return MemberResponseCode.buildReturnMap(MemberResponseCode.RESPONSE_CODE_USER_VALIDATE_NO_SUCCESS, false);
}
if(isValidate){
//HttpSession session =request.getSession(false);
Login login = new Login(user.getId(), user.getLoginName(), user.getPassword());
//給用戶jwt加密生成token
String token = JWT.sign(login, 60L* 1000L* 30L);
Map result =new HashMap();
result.put("loginToken", token);
result.put("userId", user.getId());
result.put("user", user);
//保存用戶信息到session
//session.setAttribute(user.getId() + "@@" + token, user);
//重建用戶信息
this.rebuildLoginUser(user.getId(), token);
return ResponseCode.buildReturnMap(ResponseCode.RESPONSE_CODE_LOGIN_SUCCESS, result);
}
return ResponseCode.buildReturnMap(ResponseCode.USER_LOGIN_PWD_ERROR, false);
}
9. 測(cè)試sso單點(diǎn)登錄:
返回結(jié)果集:
Java代碼
{
"message":"用戶登錄成功",
"data":?{
"loginToken":"eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9.eyJleHAiOjE1MDkzODA1OTU0NTksInBheWxvYWQiOiJ7XCJ1aWRcIjpcIjExXCIsXCJsb2dpbk5hbWVcIjpcImFkbWluXCIsXCJwYXNzd29yZFwiOlwiZjU0NGQxM2QyY2EwNDU5ZGQ0ZTU1NzVjNmZkYWIzMzM0MzE1MWFlZjgwYmE5ZTNiN2U1ZjM2MzJcIn0ifQ.56L60WtxHXSu9vNs6XsWy5zbmc3kP_IWG1YpReK50DM",
"userId":"11",
"user":?{
"QQ":"2147775633",
"id":"11",
"isNewRecord":false,
"remarks":"",
"createDate":"2017-08-08?08:08:08",
"updateDate":"2017-10-29?11:23:50",
"loginName":"admin",
"no":"00012",
"name":"admin",
"email":"2147775633@qq.com",
"phone":"400000000",
"mobile":"13888888888",
"userType":"",
"loginIp":"0:0:0:0:0:0:0:1",
"loginDate":"2017-10-30?10:48:06",
"loginFlag":"1",
"photo":"",
"idCard":"420888888888888888",
"oldLoginIp":"0:0:0:0:0:0:0:1",
"oldLoginDate":"2017-10-30?10:48:06",
"roleNames":"",
"admin":false
}
},
"code":200
}
愿意了解框架技術(shù)或者源碼的朋友直接求求交流分享技術(shù):3133806896
分布式的一些解決方案,有愿意了解的朋友可以找我們團(tuán)隊(duì)探討