13.Mybatis是如何进行分页的?
1.直接在Select语句中增加数据库提供的分页关键字,然后在应用程序中传递当前页和每页展示页数
2.Mybatis中提供的RowBounds对象,实现内存级别分页
3.基于Mybatis里面的Interceptor拦截器,在Select语句执行之前动态拼接关键字
14Mybatis里的 # 和 $ 的区别?
Mybatis 在处理 #{} 时,会创建预编译的 SQL 语句,将 SQL 中的 #{} 替换为 ? 号,在执行 SQL 时会为预编译 SQL 中的占位符(?)赋值,调用 PreparedStatement 的 set 方法来赋值,预编译的 SQL 语句执行效率高,并且可以防止SQL 注入,提供更高的安全性,适合传递参数值。
Mybatis 在处理 ${} 时,只是创建普通的 SQL 语句,然后在执行 SQL 语句时 MyBatis 将参数直接拼入到 SQL 里,不能防止 SQL 注入,因为参数直接拼接到 SQL 语句中,如果参数未经过验证、过滤,可能会导致安全问题。
15.@Transactional注解底层原理
@Transactional
注解是Spring框架中用于声明式事务管理的核心组件之一,它通过AOP(面向切面编程)机制实现对方法或类的事务管理。以下是@Transactional
注解的底层原理及其工作机制的详细解析:
1. 声明式事务管理概述
Spring提供了两种事务管理方式:
- 编程式事务管理:通过手动编写代码来控制事务的开启、提交和回滚。
- 声明式事务管理:通过配置或注解的方式,将事务管理与业务逻辑分离,开发者只需关注业务逻辑。
@Transactional
注解属于声明式事务管理的一种,它允许开发者通过简单的注解声明一个方法或类需要事务支持,而不需要显式地编写事务管理代码。
2. @Transactional注解的工作流程
当一个方法被标注为@Transactional
时,Spring会根据注解的配置动态生成代理对象,并在代理对象中插入事务管理逻辑。以下是其工作流程:
(1) 代理对象的创建
- Spring容器在启动时会扫描所有标注了
@Transactional
的方法或类。 - 对于这些方法或类,Spring会为其创建代理对象(可以是JDK动态代理或CGLIB代理)。
-
- 如果目标类实现了接口,则使用JDK动态代理。
- 如果目标类没有实现接口,则使用CGLIB代理。
代理对象的作用是拦截对目标方法的调用,并在方法执行前后插入事务管理逻辑。
(2) 事务拦截器的执行
- 当调用标注了
@Transactional
的方法时,实际上是调用了代理对象的方法。 - 代理对象会将调用转发给
TransactionInterceptor
(事务拦截器),TransactionInterceptor
负责事务的开启、提交和回滚。 - 具体步骤如下:
-
- 开启事务:在方法执行前,
TransactionInterceptor
会根据事务配置从PlatformTransactionManager
中获取一个事务对象,并开启一个新的事务或加入现有事务。 - 执行目标方法:调用实际的目标方法。
- 提交或回滚事务:
- 开启事务:在方法执行前,
-
-
- 如果目标方法正常执行完成(未抛出异常),则提交事务。
- 如果目标方法抛出了未被捕获的异常,则根据配置决定是否回滚事务。
-
(3) 事务管理器的协作
PlatformTransactionManager
是Spring事务管理的核心接口,它定义了事务的基本操作(如getTransaction
、commit
和rollback
)。- 常见的实现类包括:
-
DataSourceTransactionManager
:用于基于JDBC的事务管理。HibernateTransactionManager
:用于基于Hibernate的事务管理。JpaTransactionManager
:用于基于JPA的事务管理。
- 事务管理器会根据具体的持久化技术与数据库交互,确保事务的一致性。
3. 总结
@Transactional
注解的底层原理可以概括为以下几点:
- 代理机制:通过动态代理(JDK动态代理或CGLIB代理)实现对目标方法的拦截。
- 事务拦截器:由
TransactionInterceptor
负责事务的开启、提交和回滚。 - 事务管理器:由
PlatformTransactionManager
负责具体的事务操作。 - 声明式配置:通过注解的方式简化事务管理,减少侵入性代码。
通过以上机制,Spring实现了对事务的透明化管理,使开发者能够专注于业务逻辑,而无需关心事务的具体实现细节。