众所周知,HTTP是无状态的协议,这意味着服务器无法确认用户的信息。于是乎,W3C就提出了:给每一个用户都发一个通行证,无论谁访问的时候都需要携带通行证,这样服务器就可以从通行证上确认用户的信息。通行证就是。

如果说是检查用户身上的”通行证“来确认用户的身份,那么就是通过检查服务器上的”客户明细表“来确认用户的身份的。相当于在服务器中建立了一份“客户明细表”。

HTTP协议是无状态的,不能依据HTTP连接来判断是否为同一个用户。于是乎:服务器向用户浏览器发送了一个名为的,它的值是的id值。其实是依据来识别是否是同一个用户。

所以,一般我们单系统实现登录会这样做:

我之前Demo的代码,可以参考一下:

<code class="java language-java hljs" style="box-sizing: border-box;padding: 0.5em;font-size: 14px;color: rgb(67, 79, 84);line-height: 18px;background-image: initial;background-position: initial;background-size: initial;background-repeat: initial;background-attachment: initial;background-origin: initial;background-clip: initial;font-family: Consolas, Inconsolata, Courier, monospace;display: block;overflow-x: auto;letter-spacing: 0px;box-shadow: rgb(221, 221, 221) 0px 0px 15px 3px;margin: 30px 8px 20px !important;overflow-wrap: normal !important;border-radius: 8px !important;word-break: normal !important;overflow-y: auto !important;"> /**
 * 用户登陆
 */

@PostMapping(value = "/user/session", produces = {"application/json;charset=UTF-8"})
public Result login(String mobileNo, String password, String inputCaptcha, HttpSession session, HttpServletResponse response) {

    //判断验证码是否正确
    if (WebUtils.validateCaptcha(inputCaptcha, "captcha", session)) {

        //判断有没有该用户
        User user = userService.userLogin(mobileNo, password);
        if (user != null) {
            /*设置自动登陆,一个星期.  将token保存在数据库中*/
            String loginToken = WebUtils.md5(new Date().toString() + session.getId());
            user.setLoginToken(loginToken);
            User user1 = userService.userUpload(user);

            session.setAttribute("user", user1);

            CookieUtil.addCookie(response,"loginToken",loginToken,604800);

            return ResultUtil.success(user1);

        } else {
            return ResultUtil.error(ResultEnum.LOGIN_ERROR);
        }
    } else {
        return ResultUtil.error(ResultEnum.CAPTCHA_ERROR);
    }

}

/**
 * 用户退出
 */

@DeleteMapping(value = "/session", produces = {"application/json;charset=UTF-8"})
public Result logout(HttpSession session,HttpServletRequest request,HttpServletResponse response ) {

    //删除session和cookie
    session.removeAttribute("user");

    CookieUtil.clearCookie(request, response, "loginToken");

    return ResultUtil.success();
}
/**
@author ozc
@version 1.0


* 拦截器;实现自动登陆功能
*/
public class UserInterceptor implements HandlerInterceptor {


@Autowired
private UserService userService;

public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object o) throws Exception {
    User sessionUser = (User) request.getSession().getAttribute("user");

    // 已经登陆了,放行
    if (sessionUser != null) {
        return true;
    } else {
        //得到带过来cookie是否存在
        String loginToken = CookieUtil.findCookieByName(request, "loginToken");
        if (StringUtils.isNotBlank(loginToken)) {
            //到数据库查询有没有该Cookie
            User user = userService.findUserByLoginToken(loginToken);
            if (user != null) {
                request.getSession().setAttribute("user", user);
                return true;
            } else {
                //没有该Cookie与之对应的用户(Cookie不匹配)
                CookieUtil.clearCookie(request, response, "loginToken");
                return false;
            }
        } else {

            //没有cookie、也没有登陆。是index请求获取用户信息,可以放行
            if (request.getRequestURI().contains("session")) {
                return true;
            }

            //没有cookie凭证
            response.sendRedirect("/login.html");
            return false;
        }
    }
}
}

总结一下上面代码的思路:

的作用是什么?和有什么区别?

和 都是用来跟踪浏览器用户身份的会话方式,但是两者的应用场景不太一样。

一般用来保存用户信息比如①我们在 中保存已经登录过得用户信息,下次访问网站的时候页面可以自动帮你登录的一些基本信息给填了;②一般的网站都会有保持登录也就是说下次你再访问网站的时候就不需要重新登录了,这是因为用户登录的时候我们可以存放了一个 Token 在 中,下次登录的时候只需要根据 Token 值来查找用户即可(为了安全考虑,重新登录一般要将 Token 重写);③登录一次网站后访问网站其他页面不需要重新登录。 的主要作用就是通过服务端记录用户的状态。典型的场景是购物车,当你要添加商品到购物车的时候,系统不知道是哪个用户操作的,因为 HTTP 协议是无状态的。服务端给特定的用户创建特定的 之后就可以标识这个用户并且跟踪这个用户了。

数据保存在客户端(浏览器端)java购物车代码, 数据保存在服务器端。

存储在客户端中,而存储在服务器上,相对来说 安全性更高。如果使用 的一些敏感信息不要写入 中,最好能将 信息加密然后使用到的时候再去服务器端解密。

三、多系统登录的问题与解决3.1 不共享问题

单系统登录功能主要是用保存用户信息来实现的,但我们清楚的是:多系统即可能有多个,而是依赖当前系统的,所以系统A的和系统B的是不共享的。

java购物车代码_javaweb购物车源代码_购物车代码实现

系统A的和系统B的是不共享的

解决系统之间不共享问题有一下几种方案:

我们可以将登录功能单独抽取出来,做成一个子系统。

java购物车代码_购物车代码实现_javaweb购物车源代码

抽取出来成为子系统

抽取出来成为子系统

SSO(登录系统)的逻辑如下:

// 登录功能(SSO单独的服务)
@Override
public TaotaoResult login(String username, String password) throws Exception {

    //根据用户名查询用户信息
    TbUserExample example = new TbUserExample();
    Criteria criteria = example.createCriteria();
    criteria.andUsernameEqualTo(username);
    List list = userMapper.selectByExample(example);
    if (null == list || list.isEmpty()) {
        return TaotaoResult.build(400"用户不存在");
    }
    //核对密码
    TbUser user = list.get(0);
    if (!DigestUtils.md5DigestAsHex(password.getBytes()).equals(user.getPassword())) {
        return TaotaoResult.build(400"密码错误");
    }
    //登录成功,把用户信息写入redis
    //生成一个用户token
    String token = UUID.randomUUID().toString();
    jedisCluster.set(USER_TOKEN_KEY + ":" + token, JsonUtils.objectToJson(user));
    //设置session过期时间
    jedisCluster.expire(USER_TOKEN_KEY + ":" + token, SESSION_EXPIRE_TIME);
    return TaotaoResult.ok(token);
}

其他子系统登录时,请求SSO(登录系统)进行登录,将返回的token写到中,下次访问时则把带上:

public TaotaoResult login(String username, String password, 
        HttpServletRequest request, HttpServletResponse response)
 
{
    //请求参数
    Map param = new HashMap();
    param.put("username", username);
    param.put("password", password);
    //登录处理
    String stringResult = HttpClientUtil.doPost(REGISTER_USER_URL + USER_LOGIN_URL, param);
    TaotaoResult result = TaotaoResult.format(stringResult);
    //登录出错
    if (result.getStatus() != 200) {
        return result;
    }
    //登录成功后把取token信息,并写入cookie
    String token = (String) result.getData();
    //写入cookie
    CookieUtils.setCookie(request, response, "TT_TOKEN", token);
    //返回成功
    return result;

}

总结:

到这里,其实我们会发现其实就两个变化:

3.2 跨域的问题

上面我们解决了不能共享的问题,但其实还有另一个问题。是不能跨域的

比如说,我们请求时,浏览器会自动把的带过去给的服务器,而不会把的带过去给的服务器。

这就意味着,由于域名不同,用户向系统A登录后,系统A返回给浏览器的,用户再请求系统B的时候不会将系统A的带过去。

针对存在跨域问题,有几种解决方案:

服务端将写到客户端后,客户端对进行解析,将Token解析出来,此后请求都把这个Token带上就行了

多个域名共享,在写到客户端的时候设置的。

将Token保存在中(不依赖就没有跨域的问题了)

到这里,我们已经可以实现单点登录了。

3.3 CAS原理

说到单点登录,就肯定会见到这个名词:CAS ( ),下面说说CAS是怎么搞的。

如果已经将登录单独抽取成系统出来,我们还能这样玩。现在我们有两个系统,分别是和,一个SSO

现在我们有三个系统

首先,用户想要访问系统A受限的资源(比如说购物车功能,购物车功能需要登录后才能访问),系统A发现用户并没有登录,于是重定向到sso认证中心,并将自己的地址作为参数。请求的地址如下:

sso认证中心发现用户未登录,将用户引导至登录页面,用户进行输入用户名和密码进行登录,用户与认证中心建立全局会话(生成一份Token,写到中java购物车代码,保存在浏览器上)

4步过程

随后,认证中心重定向回系统A,并把Token携带过去给系统A,重定向的地址如下:

接着,系统A去sso认证中心验证这个Token是否正确,如果正确,则系统A和用户建立局部会话(创建)。到此,系统A和用户已经是登录状态了。

第五步和第六步

此时,用户想要访问系统B受限的资源(比如说订单功能,订单功能需要登录后才能访问),系统B发现用户并没有登录,于是重定向到sso认证中心,并将自己的地址作为参数。请求的地址如下:

注意,因为之前用户与认证中心已经建立了全局会话(当时已经把保存到浏览器上了),所以这次系统B重定向到认证中心是可以带上的。

认证中心根据带过来的发现已经与用户建立了全局会话了,认证中心重定向回系统B,并把Token携带过去给系统B,重定向的地址如下:

接着,系统B去sso认证中心验证这个Token是否正确,如果正确,则系统B和用户建立局部会话(创建)。到此,系统B和用户已经是登录状态了。

系统B的流程图

看到这里,其实SSO认证中心就类似一个中转站。

参考


限时特惠:
本站持续每日更新海量各大内部创业课程,一年会员仅需要98元,全站资源免费下载
点击查看详情

站长微信:Jiucxh

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注