【第6章】SpringBoot实战篇之登出接口


前言

上一章节我们新增了用户登录的接口,因为JWT是无状态的,对应的用户登出接口需要我们增加对应的业务逻辑去控制,这里我们使用Map简单存储用户登录信息。


一、后端代码

这里我们使用loginUsers属性来保存用户登录信息,拦截器验证部分稍有改动。

1. LoginInceptor

package org.example.springboot3.bigevent.inceptors;

import com.fasterxml.jackson.databind.ObjectMapper;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import org.example.springboot3.bigevent.entity.Result;
import org.example.springboot3.bigevent.login.LoginStorage;
import org.example.springboot3.bigevent.utils.JwtUtils;
import org.example.springboot3.bigevent.utils.ThreadLocalUtil;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.http.HttpStatus;
import org.springframework.stereotype.Component;
import org.springframework.web.servlet.HandlerInterceptor;
import java.util.Map;

/**
 * Create by zjg on 2024/5/26
 */
@Component
public class LoginInceptor implements HandlerInterceptor {
    @Autowired
    LoginStorage loginStorage;
    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        String token = request.getHeader("Authorization");
        if(token!=null&&token.contains("Bearer")){
            String tokenStr = token.substring(token.indexOf("Bearer") + 7);
            boolean verify = JwtUtils.verify(tokenStr);
            if(verify){//此处解析loginUsers,验证用户已登录
                Map<String, Object> claims = JwtUtils.getClaims(tokenStr);
                if(tokenStr.equals(loginStorage.get(claims.get("userId").toString()))){
                    ThreadLocalUtil.set(claims);//用户信息放置ThreadLocal
                    return true;
                };
            }
        }
        response.setStatus(HttpStatus.UNAUTHORIZED.value());
        response.setContentType("application/json;charset=UTF-8");
        ObjectMapper objectMapper = new ObjectMapper();
        objectMapper.writerFor(Result.class);
        String message = objectMapper.writeValueAsString(Result.error("token验证失败,请重新获取token后重试!"));
        response.getWriter().println(message);
        return false;
    }

    @Override
    public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
        ThreadLocalUtil.remove();
    }

}


2. UserController1

登出接口需要携带token信息,因为只有登录了才能登出且必须和登录时保存的token信息一致,才能成功。

@Autowired
LoginStorage loginStorage;
/**
* 登出
 * @param loginUser 登录用户信息
 * @param token token
 * @return Result
 */
@RequestMapping("logout")
public Result logout(@Valid User loginUser,@RequestHeader("Authorization") String token){
    String message="用户名/密码不正确";
    User user = userSerivce.findUserByName(loginUser.getUsername());
    if(user!=null){//用户存在
        if(token!=null&&token.contains("Bearer")){
            String tokenStr = token.substring(token.indexOf("Bearer") + 7);
            boolean verify = JwtUtils.verify(tokenStr);
            if(verify&&tokenStr.equals(loginStorage.get(user.getId().toString()))){
                loginStorage.remove(user.getId().toString());
                return Result.success("登出成功");
            }
        }
    }
    return Result.error(message);
}

二、测试

1.登录

在这里插入图片描述

2.查询

在这里插入图片描述

3.登出

在这里插入图片描述

4.再次查询

在这里插入图片描述


总结

回到顶部

这种方式适合单机部署,集群部署会导致登录信息无法共享,可将Map值存储到redis中。

好的,针对您的问题,我可以给您提供一些基本的思路和代码示例。 首先,您需要在Spring Boot项目中集成MyBatis,可以通过引入相应的依赖来实现。然后,您需要定义一个用户实体类,并在MyBatis中定义相应的Mapper接口和SQL语句,以实现用户的登录和登出功能。 以下是一个示例用户实体类的代码: ```java public class User { private int id; private String username; private String password; // 省略getter和setter方法 } ``` 接下来是一个示例Mapper接口的代码: ```java @Mapper public interface UserMapper { @Select("SELECT * FROM user WHERE username = #{username} AND password = #{password}") User login(@Param("username") String username, @Param("password") String password); @Update("UPDATE user SET token = NULL WHERE id = #{id}") void logout(@Param("id") int id); } ``` 上面的代码中,login()方法用于实现用户登录功能,通过传入用户名和密码来查询数据库中是否存在该用户。logout()方法用于实现用户登出功能,将用户的token字段设为NULL。 最后,您可以在Controller层中定义相应的请求处理方法来调用Mapper接口中的方法,以实现用户登录和登出功能。以下是一个示例Controller层的代码: ```java @RestController public class UserController { @Autowired private UserMapper userMapper; @PostMapping("/login") public String login(@RequestParam String username, @RequestParam String password) { User user = userMapper.login(username, password); if (user != null) { // 登录成功,生成token并返回 String token = UUID.randomUUID().toString(); user.setToken(token); // 将token保存到数据库中 userMapper.updateToken(user.getId(), token); return token; } else { // 登录失败,返回错误提示 return "用户名或密码错误"; } } @PostMapping("/logout") public String logout(@RequestParam int id) { userMapper.logout(id); return "登出成功"; } } ``` 上面的代码中,login()方法用于处理登录请求,如果登录成功,生成一个随机的token并保存到数据库中,然后将token返回给客户端。logout()方法用于处理登出请求,将用户的token字段设为NULL。 以上就是一个简单的Spring Boot + MyBatis实现用户登录和登出功能的示例,希望对您有所帮助。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值