①Nginx反向代理
概念:
优点:
反向代理配置方式:
负载均衡配置方式:
②MD5加密,使用Hutool工具
③ThreadLocal
封装线程操作的类:
//线程操作封装类
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();
}
}
只需要在拦截器中拦截的时候把数据放到线程中,在之后的Controller层和Service层中都可以直接在线程中取数据
④使用Mybatis-plus实现分页查询
mpConfig.java
import com.baomidou.mybatisplus.extension.plugins.MybatisPlusInterceptor;
import com.baomidou.mybatisplus.extension.plugins.inner.PaginationInnerInterceptor;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
@Configuration
public class mpConfig {
@Bean
public MybatisPlusInterceptor mybatisPlusInterceptor(){
//定义分页查询拦截器
MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();
//添加拦截器
interceptor.addInnerInterceptor(new PaginationInnerInterceptor());
return interceptor;
}
}
service层:
public PageResult pageQuery(EmployeePageQueryDTO employeePageQueryDTO) {
//1.创建Ipage对象
IPage page = new Page(employeePageQueryDTO.getPage(),employeePageQueryDTO.getPageSize());
//2.Mapper层对象去进行查询
//2.1 wapper对象去条件查询
QueryWrapper wrapper = new QueryWrapper();
wrapper.likeRight("name",employeePageQueryDTO.getName());
//2.2 如果没输入模糊查询
if(employeePageQueryDTO.getName() == null){
employeeMapper.selectPage(page,null);
}
//2.3 如果模糊查询
else
employeeMapper.selectPage(page,wrapper);
//3. 此时Page中已经存了我们所要的数据
//显示第几页
System.out.println(page.getCurrent());
//显示每页大小
System.out.println(page.getSize());
//显示一共有多少页
System.out.println(page.getPages());
//显示一共用多少条数据
System.out.println(page.getTotal());
//显示该页的数据
System.out.println(page.getRecords());
//返回我们所需的数据
return new PageResult(page.getTotal(),page.getRecords());
}
⑤有关日期的显示格式的统一
第一种方式:添加@JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss")注解
第二种方式:全局配置一下
/**
* 对象映射器:基于jackson将Java对象转为json,或者将json转为Java对象
* 将JSON解析为Java对象的过程称为 [从JSON反序列化Java对象]
* 从Java对象生成JSON的过程称为 [序列化Java对象到JSON]
*/
public class JacksonObjectMapper extends ObjectMapper {
public static final String DEFAULT_DATE_FORMAT = "yyyy-MM-dd";
//public static final String DEFAULT_DATE_TIME_FORMAT = "yyyy-MM-dd HH:mm:ss";
public static final String DEFAULT_DATE_TIME_FORMAT = "yyyy-MM-dd HH:mm";
public static final String DEFAULT_TIME_FORMAT = "HH:mm:ss";
public JacksonObjectMapper() {
super();
//收到未知属性时不报异常
this.configure(FAIL_ON_UNKNOWN_PROPERTIES, false);
//反序列化时,属性不存在的兼容处理
this.getDeserializationConfig().withoutFeatures(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES);
SimpleModule simpleModule = new SimpleModule()
.addDeserializer(LocalDateTime.class, new LocalDateTimeDeserializer(DateTimeFormatter.ofPattern(DEFAULT_DATE_TIME_FORMAT)))
.addDeserializer(LocalDate.class, new LocalDateDeserializer(DateTimeFormatter.ofPattern(DEFAULT_DATE_FORMAT)))
.addDeserializer(LocalTime.class, new LocalTimeDeserializer(DateTimeFormatter.ofPattern(DEFAULT_TIME_FORMAT)))
.addSerializer(LocalDateTime.class, new LocalDateTimeSerializer(DateTimeFormatter.ofPattern(DEFAULT_DATE_TIME_FORMAT)))
.addSerializer(LocalDate.class, new LocalDateSerializer(DateTimeFormatter.ofPattern(DEFAULT_DATE_FORMAT)))
.addSerializer(LocalTime.class, new LocalTimeSerializer(DateTimeFormatter.ofPattern(DEFAULT_TIME_FORMAT)))
.addSerializer(BigInteger.class, ToStringSerializer.instance)
.addSerializer(Long.class, ToStringSerializer.instance);
//注册功能模块 例如,可以添加自定义序列化器和反序列化器
this.registerModule(simpleModule);
}
}
/**
* 配置类,注册web层相关组件
*/
@Configuration
@Slf4j
public class WebMvcConfiguration extends WebMvcConfigurationSupport {
/**
* 扩展Spring MVC框架的消息转换器
* @param converters
*/
@Override
protected void extendMessageConverters(List<HttpMessageConverter<?>> converters) {
log.info("扩展消息转换器");
//创建一个消息转换器对象
MappingJackson2HttpMessageConverter converter = new MappingJackson2HttpMessageConverter();
//为消息转换器设置一个对象转换器,对象转换器可以将Java对象序列化为json数据
converter.setObjectMapper(new JacksonObjectMapper());
//将消息转换器加入容器中
converters.add(0,converter);
}
}
⑥雪花算法会导致前后端数据精度丢失
解决方法一:加@JsonSerialize(using = ToStringSerializer.class)注解
解决方法二:引入工具类
/**
* 对象映射器:基于jackson将Java对象转为json,或者将json转为Java对象
* 将JSON解析为Java对象的过程称为 [从JSON反序列化Java对象]
* 从Java对象生成JSON的过程称为 [序列化Java对象到JSON]
*/
public class JacksonObjectMapper extends ObjectMapper {
public static final String DEFAULT_DATE_FORMAT = "yyyy-MM-dd";
//public static final String DEFAULT_DATE_TIME_FORMAT = "yyyy-MM-dd HH:mm:ss";
public static final String DEFAULT_DATE_TIME_FORMAT = "yyyy-MM-dd HH:mm";
public static final String DEFAULT_TIME_FORMAT = "HH:mm:ss";
public JacksonObjectMapper() {
super();
//收到未知属性时不报异常
this.configure(FAIL_ON_UNKNOWN_PROPERTIES, false);
//反序列化时,属性不存在的兼容处理
this.getDeserializationConfig().withoutFeatures(DeserializationFeature.FAIL_ON_UNKNOWN_PROPERTIES);
SimpleModule simpleModule = new SimpleModule()
.addDeserializer(LocalDateTime.class, new LocalDateTimeDeserializer(DateTimeFormatter.ofPattern(DEFAULT_DATE_TIME_FORMAT)))
.addDeserializer(LocalDate.class, new LocalDateDeserializer(DateTimeFormatter.ofPattern(DEFAULT_DATE_FORMAT)))
.addDeserializer(LocalTime.class, new LocalTimeDeserializer(DateTimeFormatter.ofPattern(DEFAULT_TIME_FORMAT)))
.addSerializer(LocalDateTime.class, new LocalDateTimeSerializer(DateTimeFormatter.ofPattern(DEFAULT_DATE_TIME_FORMAT)))
.addSerializer(LocalDate.class, new LocalDateSerializer(DateTimeFormatter.ofPattern(DEFAULT_DATE_FORMAT)))
.addSerializer(LocalTime.class, new LocalTimeSerializer(DateTimeFormatter.ofPattern(DEFAULT_TIME_FORMAT)))
//在这里是解决雪花算法导致前后端精度丢失
.addSerializer(BigInteger.class, ToStringSerializer.instance)
.addSerializer(Long.class, ToStringSerializer.instance);
//注册功能模块 例如,可以添加自定义序列化器和反序列化器
this.registerModule(simpleModule);
}
}
/**
* 配置类,注册web层相关组件
*/
@Configuration
@Slf4j
public class WebMvcConfiguration extends WebMvcConfigurationSupport {
/**
* 扩展Spring MVC框架的消息转换器
* @param converters
*/
@Override
protected void extendMessageConverters(List<HttpMessageConverter<?>> converters) {
log.info("扩展消息转换器");
//创建一个消息转换器对象
MappingJackson2HttpMessageConverter converter = new MappingJackson2HttpMessageConverter();
//为消息转换器设置一个对象转换器,对象转换器可以将Java对象序列化为json数据
converter.setObjectMapper(new JacksonObjectMapper());
//将消息转换器加入容器中
converters.add(0,converter);
}
}
⑦对象的拷贝BeanUtil.copyProperties(old,new)
public void updateEm(EmployeeDTO employeeDTO) {
Employee employee = new Employee();
BeanUtils.copyProperties(employeeDTO,employee);
employee.setUpdateTime(LocalDateTime.now());
employee.setUpdateUser(BaseContext.getCurrentId());
updateById(employee);
}
⑧自己打代码学到了一个知识点
就是wapper的判断条件:因为不知道前端传来的对象中是否有模糊查询,所以之前写的时候就在外面加判断条件,十分不美观。
//2.1 wapper对象去条件查询
QueryWrapper wrapper = new QueryWrapper();
wrapper.likeRight("name",employeePageQueryDTO.getName());
//2.2 如果没输入模糊查询
if(employeePageQueryDTO.getName() == null){
employeeMapper.selectPage(page,null);
}
//2.3 如果模糊查询
else {
employeeMapper.selectPage(page, wrapper);
}
在看like以及eq的源码后发现其实他第一个参数也是可以给一个判断条件,改进:
QueryWrapper wrapper = new QueryWrapper();
wrapper.like(employeePageQueryDTO.getName() != null,"name",employeePageQueryDTO.getName());
employeeMapper.selectPage(page, wrapper);