当前位置: 首页 > news >正文

苍穹外卖Day2

一、DIgestUtil实现md5加密

1.1为什么要对数据进行加密

常用方法:
`DigestUtils`是Apache Commons Codec库提供的一个实用工具类,用于处理摘要算法(如MD5、SHA等)的相关操作。其中包括了一些用于MD5加密的函数。以下是一些`DigestUtils`中有关MD5加密的函数:

使用案例

实战:用户注册时候将注册的密码通过md5加密后上传&&用户登陆时候自动将密码进行md5加密再进行对比

password = DigestUtils.md5DigestAsHex(password.getBytes());  
  
if (!password.equals(employee.getPassword())) {  
    //密码错误  
    throw new PasswordErrorException(MessageConstant.PASSWORD_ERROR);  
}  
  
if (employee.getStatus() == StatusConstant.DISABLE) {  
    //账号被锁定  
    throw new AccountLockedException(MessageConstant.ACCOUNT_LOCKED);  
}  
  
//3、返回实体对象  
return employee;

二、介绍并配置使用Swagger

2.1 介绍Swagger

Swagger是一种用于设计、构建、文档化和消费RESTful Web服务的开源工具集。它的主要目标是简化API的开发和维护流程,提高团队协作效率,同时提供一致且易于理解的API文档。

2.2 配置Swagger

在pom.xml中导入依赖

<dependency>
   <groupId>com.github.xiaoymin</groupId>
   <artifactId>knife4j-spring-boot-starter</artifactId>
   <version>3.0.2</version>
</dependency>

配置类配置,在config包下WebMvc下进行配置

   /**
     * 通过knife4j生成接口文档
     * @return
     */
    @Bean
    public Docket docket1() {
        log.info("开始生成接口文档...");
        ApiInfo apiInfo = new ApiInfoBuilder()
                .title("苍穹外卖项目接口文档")
                .version("2.0")
                .description("苍穹外卖项目接口文档")
                .build();
        Docket docket = new Docket(DocumentationType.SWAGGER_2)
                .groupName("用户端")
                .apiInfo(apiInfo)
                .select()
                //指定要扫描的包
                .apis(RequestHandlerSelectors.basePackage("com.sky.controller.user"))
                .paths(PathSelectors.any())
                .build();
        return docket;
    }

    /**
     * 设置静态资源映射
     * @param registry
     */
    protected void addResourceHandlers(ResourceHandlerRegistry registry) {
        log.info("开始设置静态资源映射...");
        registry.addResourceHandler("/doc.html").addResourceLocations("classpath:/META-INF/resources/");
        registry.addResourceHandler("/webjars/**").addResourceLocations("classpath:/META-INF/resources/webjars/");
    }

注:
1.如果缺少

registry.addResourceHandler("/doc.html").addResourceLocations("classpath:/META-INF/resources/"); 会报错404
2.如果缺少          //__指定生成接口需要扫描的包      .apis(RequestHandlerSelectors.basePackage("com.sky.controller")) 则无法找到对应类 

 2.3 使用Swagger

在运行后端后,在端口后面加上doc.html即可跳转到对应的页面

Swagger常用的注解配置

使用案例:

1.@Api

//  添加@Api注解,描述员工管理接口  
@Api(tags = "员工管理")  
public class EmployeeController {
  。。。业务。。。
}

2.@ApiOperation

@ApiOperation("员工登录")  
@PostMapping("/login")  
public Result<EmployeeLoginVO> login(@RequestBody EmployeeLoginDTO employeeLoginDTO) {  
    log.info("员工登录:{}", employeeLoginDTO);  
  
    Employee employee = employeeService.login(employeeLoginDTO);  
  
    //登录成功后,生成jwt令牌  
    Map<String, Object> claims = new HashMap<>();  
    claims.put(JwtClaimsConstant.EMP_ID, employee.getId());  
    String token = JwtUtil.createJWT(  
            jwtProperties.getAdminSecretKey(),  
            jwtProperties.getAdminTtl(),  
            claims);  
  
    EmployeeLoginVO employeeLoginVO = EmployeeLoginVO.builder()  
            .id(employee.getId())  
            .userName(employee.getUsername())  
            .name(employee.getName())  
            .token(token)  
            .build();  
  
    return Result.success(employeeLoginVO);  
}

3.@ApiModel与@ApiModelProperty

@Data  
@ApiModel(description = "员工登录时传递的数据模型")  
public class EmployeeLoginDTO implements Serializable {  
  
    @ApiModelProperty("用户名")  
    private String username;  
  
    @ApiModelProperty("密码")  
    private String password;  
  
}

 效果如下:

思考:有swaggar还需要像APIFOx(PostMan)这种工具吗 

由于开发阶段前端和后端是并行开发的,后端完成某个功能后,此时前端对应的功能可能还没有开发完成
导致无法进行前后端联调测试。所以在开发阶段,后端测试主要以接口文档测试为主。


三、业务逻辑开发

3.1 新增并测试员工

3.2 对应的DTO

三层架构代码:
1.控制层Controller层:

@PostMapping("/save")  
@ApiOperation("员工注册")  
public Result save(EmployeeDTO employeeDTO){  
    log.info("员工注册:{}", employeeDTO);  
    employeeService.save(employeeDTO);  
    return Result.success();  
}

2.服务层Service层:
 

void save(EmployeeDTO employeeDTO);

Service具体实现

// 保存员工信息  
public void save(EmployeeDTO employeeDTO){  
    Employee employee = new Employee();  
    //通过BeanUtils.copyProperties()方法将employeeDTO中的属性值复制到employee中  
    BeanUtils.copyProperties(employeeDTO, employee);  
    //将employee剩下的属性进行赋值  
  
    //1.对默认密码进行MD5加密  
    employee.setPassword(DigestUtils.md5DigestAsHex(PasswordConstant.DEFAULT_PASSWORD.getBytes()));  
    //2.设置员工状态为启用  
    employee.setStatus(StatusConstant.ENABLE);  
  
    //3.设置创建时间和更新时间  
    employee.setCreateTime(LocalDateTime.now());  
    employee.setUpdateTime(LocalDateTime.now());  
  
    //4.设置员工创建人和更新人id  
    //TODO 从session中获取当前登录员工的id  
    employee.setCreateUser(10L);  
    employee.setUpdateUser(10L);  
  
    //5.调用employeeMapper.save()方法保存员工信息  
    employeeMapper.save(employee);  
};

Mapper层

/**  
 * 插入新员工  
 * @param employee  
 */  
@Insert("insert into sky_take_out.employee(name, username, password, phone, sex, id_number, create_time, " +  
        "update_time, create_user, update_user)"+  
        "values" +"(#{name}, #{username}, #{password},#{phone}, #{sex}, #{idNumber}, #{createTime}, #{updateTime}, #{createUser}, #{updateUser})"  
)  
void save(Employee employee);

application.yml中开启驼峰命名

通过驼峰命名使得id_number对应上idNumber

 通过swagger测试

前后端联调测试


 四、全局异常处理

测试重复插入同个id的员工

 这里就进行了报错,并且并未对报错的内容进行处理。比如重复插入时候应该抛出一个异常告诉前端,请勿重复插入

 创建全局异常处理类

核心注解:

@ExceptionHandler 

/**  
 * 捕获SqlIntegrityConstraintViolationException异常  
 * @return  
 */  
@ExceptionHandler  
public Result exceptionHander(SQLIntegrityConstraintViolationException ex){  
    //Duplicate entry 'zhangsan' for key 'employee.idx_username'异常信息案例  
    String message = ex.getMessage();  
    //判断异常信息中是否包含“Duplicate entry”关键字  
    if(message.contains("Duplicate entry"))  
    {  
        //动态提取出重复的数据  
        String[] split = message.split(" ");  
        String username = split[2];  
        String msg = username+"数据已存在,不能重复添加";  
        return Result.error(msg);  
    }  
    //未知错误  
    return Result.error(MessageConstant.UNKNOWN_ERROR);  
  
}

五、通过ThreadLocal存放用户信息

 每次用户登录时候,应该创建一个token。将用户的个人信息和有效期等存储存起来。用户每次操作时候就可以通过访问当前token来得知当前用户和token是否过期。

ThreadLocal常用方法:
`ThreadLocal` 在 Java 中是一个非常有用的类,主要用于维护变量在使用线程中的线程局部性,即每个线程都有一个变量的单独副本。这样可以确保线程之间的数据隔离,避免了多线程环境下的同步问题。以下是 `ThreadLocal` 类中一些常用的方法:

1.void set(T value)
    设置当前线程的线程局部变量的值。每个线程调用此方法时,都只会影响到调用线程中存储的副本。
2. T get()
    返回当前线程所对应的线程局部变量。如果当前线程之前没有设置过这个变量的值,`ThreadLocal` 可能会初始化这个变量并返回初始值。
3.void remove()
    移除当前线程的局部变量,如果之后还需要使用同一个变量,`ThreadLocal` 可能会重新进行初始化。
4.T initialValue()
    返回该线程局部变量的初始值。这个方法是一个被 `protected` 修饰的方法,通常用于通过匿名内部类覆盖以提供初始值。默认情况下,`initialValue()` 方法返回 `null`。

使用案例:
 

public class BaseContext {  
  
    public static ThreadLocal<Long> threadLocal = new ThreadLocal<>();  
  
    public static void setCurrentId(Long id) {  
        threadLocal.set(id);  
    }  
  
    public static Long getCurrentId() {  
        return threadLocal.get();  
    }  
  
    public static void removeCurrentId() {  
        threadLocal.remove();  
    }  
  
}

六、配置JWT并使用拦截器

通过JWT工具类快速上手该技术

public class JwtUtil {  
    /**  
     * 生成jwt  
     * 使用Hs256算法, 私匙使用固定秘钥  
     *  
     * @param secretKey jwt秘钥  
     * @param ttlMillis jwt过期时间(毫秒)  
     * @param claims    设置的信息  
     * @return  
     */  
    public static String createJWT(String secretKey, long ttlMillis, Map<String, Object> claims) {  
        // 指定签名的时候使用的签名算法,也就是header那部分  
        SignatureAlgorithm signatureAlgorithm = SignatureAlgorithm.HS256;  
  
        // 生成JWT的时间  
        long expMillis = System.currentTimeMillis() + ttlMillis;  
        Date exp = new Date(expMillis);  
  
        // 设置jwt的body  
        JwtBuilder builder = Jwts.builder()  
                // 如果有私有声明,一定要先设置这个自己创建的私有的声明,这个是给builder的claim赋值,一旦写在标准的声明赋值之后,就是覆盖了那些标准的声明的  
                .setClaims(claims)  
                // 设置签名使用的签名算法和签名使用的秘钥  
                .signWith(signatureAlgorithm, secretKey.getBytes(StandardCharsets.UTF_8))  
                // 设置过期时间  
                .setExpiration(exp);  
  
        return builder.compact();  
    }  
  
    /**  
     * Token解密  
     *  
     * @param secretKey jwt秘钥 此秘钥一定要保留好在服务端, 不能暴露出去, 否则sign就可以被伪造, 如果对接多个客户端建议改造成多个  
     * @param token     加密后的token  
     * @return  
     */  
    public static Claims parseJWT(String secretKey, String token) {  
        // 得到DefaultJwtParser  
        Claims claims = Jwts.parser()  
                // 设置签名的秘钥  
                .setSigningKey(secretKey.getBytes(StandardCharsets.UTF_8))  
                // 设置需要解析的jwt  
                .parseClaimsJws(token).getBody();  
        return claims;  
    }  
  
}

Jwt配置类:

@Component  
@ConfigurationProperties(prefix = "sky.jwt")  
@Data  
public class JwtProperties {  
  
    /**  
     * 管理端员工生成jwt令牌相关配置  
     */  
    private String adminSecretKey;  
    private long adminTtl;  
    private String adminTokenName;  
  
    /**  
     * 用户端微信用户生成jwt令牌相关配置  
     */  
    private String userSecretKey;  
    private long userTtl;  
    private String userTokenName;  
  
}

jwt管理员和员工的yml配置

sky:  
  jwt:  
    # 设置jwt签名加密时使用的秘钥  
    admin-secret-key: itcast  
    # 设置jwt过期时间  
    admin-ttl: 7200000  
    # 设置前端传递过来的令牌名称  
    admin-token-name: token  
    #  设置用户的jwt签名加密时使用的秘钥  
    user-secret-key: alphaMilk  
    # 设置用户的jwt过期时间  
    user-ttl: 7200000  
    # 设置用户的jwt签名加密时使用的秘钥  
    user-token-name: authentication

配置拦截器

/**  
 * jwt令牌校验的拦截器  
 */  
@Component  
@Slf4j  
public class JwtTokenAdminInterceptor implements HandlerInterceptor {  
  
    @Autowired  
    private JwtProperties jwtProperties;  
  
    /**  
     * 校验jwt  
     *     * @param request  
     * @param response  
     * @param handler  
     * @return  
     * @throws Exception  
     */    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {  
        //判断当前拦截到的是Controller的方法还是其他资源  
        if (!(handler instanceof HandlerMethod)) {  
            //当前拦截到的不是动态方法,直接放行  
            return true;  
        }  
  
        //1、从请求头中获取令牌  
        String token = request.getHeader(jwtProperties.getAdminTokenName());  
  
        //2、校验令牌  
        try {  
            log.info("jwt校验:{}", token);  
            Claims claims = JwtUtil.parseJWT(jwtProperties.getAdminSecretKey(), token);  
            Long empId = Long.valueOf(claims.get(JwtClaimsConstant.EMP_ID).toString());  
            log.info("当前员工id:", empId);  
  
            //通过ThreadLocal保存员工id  
            BaseContext.setCurrentId(empId);  
            //3、通过,放行  
            return true;  
        } catch (Exception ex) {  
            //4、不通过,响应401状态码  
            response.setStatus(401);  
            return false;  
        }  
    }  
}

将拦截器应用到配置中,在WebMvcConfig文件增加以下配置:

    @Autowired  
    private JwtTokenAdminInterceptor jwtTokenAdminInterceptor;  
  
    // 注入用户拦截器  
    @Autowired  
    private JwtTokenUserInterceptor jwtTokenUserInterceptor;  
  
    /**  
     * 注册自定义拦截器  
     *  
     * @param registry  
     */  
    protected void addInterceptors(InterceptorRegistry registry) {  
        log.info("开始注册自定义拦截器...");  
//          管理员拦截器  
        registry.addInterceptor(jwtTokenAdminInterceptor)  
                .addPathPatterns("/admin/**")  
                .excludePathPatterns("/admin/employee/login");  
//        用户拦截器  
        registry.addInterceptor(jwtTokenUserInterceptor)  
                .addPathPatterns("/user/**")  
                .excludePathPatterns("/user/user/login")  
                .excludePathPatterns("/user/shop/status");  
    }

如此就能正常进行拦截并设置拦截范围


相关文章:

  • 生物科技 网站模板下载苏州关键词排名系统
  • 网站建设程序做哪些南昌seo专业团队
  • 华侨大学英文网站建设搜索引擎营销的实现方法有
  • 武汉做旅游教育的公司网站淘宝关键词指数查询
  • 网站如何做电脑销售电商平台运营
  • 东莞 网站建设多少钱灰色关键词排名优化
  • 文件操作(C语言)
  • 蓝桥云客---蓝桥速算
  • 网络安全L2TP实验
  • 对状态模式的理解
  • 14.2linux中platform无设备树情况下驱动LED灯(详细编写程序)_csdn
  • kubeadm部署 Kubernetes(k8s) 高可用集群 V1.28.2
  • 日志统计(双指针)
  • Chrome开发者工具实战:调试三剑客
  • 使用ZYNQ芯片和LVGL框架实现用户高刷新UI设计系列教程(第六讲)
  • 新版pycharm如何实现debug调试需要参数的python文件
  • 【CSS】样式与效果
  • C语言之编译和debug工具
  • C++模板递归结构详解和使用
  • React中类组件的生命周期
  • 【51单片机】2-8【I/O口】数码管显示矩阵按键值
  • python通过调用海康SDK打开工业相机(全流程)
  • 论文修改时有哪些需要注意的问题?
  • Leedcode刷题 | Day23_回溯算法02
  • Latex入门之超详细的Latex下载安装教程
  • OpenCV图像处理实战全解析:镜像、缩放、矫正、水印与降噪技术详解