目录
Mybatis分页插件原理?
ThreadLocal?
树形表的标记字段是什么?如何查询MySQL树形表?
Mybatis的ResultType和ResultMap的区别?
#{}和${}有什么区别?
系统如何处理异常?
首先分页参数放到ThreadLocal中,拦截执行得到sql,根据数据库类型添加对应的分页语句将重写sql。例如:(select * from table where a)转换为 (select count(*) from table where a)和(select * from table where a limit ,)计算出了total总条数,pageNum当前第几页,pageSize每页大小和当前页的数据,是否为首页,是否为尾页,总页数等。
ThreadLocal叫做线程变量,意思是ThreadLocal中填充的变量属于当前线程,该变量对其他线程而言是隔离的,也就是说该变量是当前线程独有的变量。ThreadLocal为变量在每个线程中都创建了一个副本,那么每个线程可以访问自己内部的副本变量。
ThreadLoal 变量,线程局部变量,同一个 ThreadLocal 所包含的对象,在不同的 Thread 中有不同的副本。这里有几点需要注意:
ThreadLocal 提供了线程本地的实例。它与普通变量的区别在于,每个使用该变量的线程都会初始化一个完全独立的实例副本。ThreadLocal 变量通常被private static修饰。当一个线程结束时,它所使用的所有 ThreadLocal 相对的实例副本都可被回收。
总的来说,ThreadLocal 适用于每个线程需要自己独立的实例且该实例需要在多个方法中被使用,也即变量在线程间隔离而在方法或类间共享的场景
ThreadLocal
但是ThreadLocal与synchronized有本质的区别:
1、Synchronized用于线程间的数据共享,而ThreadLocal则用于线程间的数据隔离。
2、Synchronized是利用锁的机制,使变量或代码块在某一时该只能被一个线程访问。而ThreadLocal为每一个线程都提供了变量的副本
使得每个线程在某一时间访问到的并不是同一个对象,这样就隔离了多个线程对数据的数据共享。
而Synchronized却正好相反,它用于在多个线程间通信时能够获得数据共享。
一句话理解ThreadLocal,threadlocl是作为当前线程中属性ThreadLocalMap集合中的某一个Entry的key值Entry(threadlocl,value),虽然不同的线程之间threadlocal这个key值是一样,但是不同的线程所拥有的ThreadLocalMap是独一无二的,也就是不同的线程间同一个ThreadLocal(key)对应存储的值(value)不一样,从而到达了线程间变量隔离的目的,但是在同一个线程中这个value变量地址是一样的。
以上关于ThreadLocal转载自史上最强ThreadLocal详解
标记字段是parentid即父节点的id。
查询一个树形表的方法:
1)当层级固定时可以用表的自连接查询
2)如果想灵活查询每个层级使用MySQL递归的方法,使用with RECURSIVE实现
ResultType:指定映射类型,只要查询的字段名和类型的属性名匹配可以自动映射
ResultMap:自定义映射规则,当查询的字段名和映射类型的属性不匹配时可以通过ResultMap自定义映射规则,也可以实现一对多,一对一映射。
#{}是一个占位符,可以防止sql注入。
${}用在动态sql拼接字符串,可能导致sql注入。
自定义一个统一的异常处理器去捕获并处理异常。
使用控制器增加注解@ControllerAdvice和异常处理注解@ExceptionHandler来实现
1)处理自定义异常
程序在编写代码时根据校验结果主动抛出自定义异常类对象,抛出异常时指定详细的异常信息,异常处理器捕获异常信息记录异常日志并响应给用户。
2)处理未知异常
接口执行过程中的一些运行时异常也会由异常处理器统一捕获,记录异常日志,统一响应给用户500错误,在异常处理器中还可以针对某个异常类型进行单独处理。
使用基于JSR303的校验框架实现,SpringBoot提供了JSR-303的支持,它就是spring-boot-starter-validation,它包括了很多校验规则,在controller方法上开启校验。业务逻辑的校验在service中进行,多个接口共用一个类型时可以开启校验分组。