您的位置:首页 > 科技 > 能源 > 网页制作教程软件_网站权重是什么意思_中国舆情观察网_站长收录平台

网页制作教程软件_网站权重是什么意思_中国舆情观察网_站长收录平台

2025/4/7 21:39:14 来源:https://blog.csdn.net/dogxixi/article/details/146798629  浏览:    关键词:网页制作教程软件_网站权重是什么意思_中国舆情观察网_站长收录平台
网页制作教程软件_网站权重是什么意思_中国舆情观察网_站长收录平台

一、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*/@Beanpublic 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");  }

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


版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com