Spring的AOP实现原理其实很简单,就是通过动态代理实现的。如果我们为Spring的某个bean配置了切面,那么Spring在创建这个bean的时候,实际上创建的是这个bean的一个代理对象,我们后续对bean中方法的调用,实际上调用的是代理类重写的代理方法。而Spring的AOP使用了两种动态代理,分别是JDK的动态代理,以及CGLib的动态代理。
1.静态代理
//接口
public interface Rent {
public void rent();
}
//真实角色类
public class Host implements Rent{
@Override
public void rent() {
System.out.println("房东要出租房子!");
}
}
//代理类
public class RentProxy implements Rent{
private Host host;
public RentProxy() {
}
public RentProxy(Host host) {
this.host = host;
}
@Override
public void rent() {
seeHouse();
host.rent();
hetong();
fare();
}
//附属操作
public void seeHouse(){
System.out.println("中介带你看房子");
}
public void hetong(){
System.out.println("签租赁合同");
}
public void fare(){
System.out.println("收中介费");
}
}
//测试类
public class Client {
public static void main(String[] args) {
Host host = new Host();
//host.rent();
//代理
RentProxy proxy = new RentProxy(host);
proxy.rent();
}
}
2.动态代理
//接口
public interface UserService {
public void add();
public void delete();
public void update();
public void query();
}
//实现类
public class UserServiceImpl implements UserService{
@Override
public void add() {
System.out.println("增加了一个用户");
}
@Override
public void delete() {
System.out.println("删除了一个用户");
}
@Override
public void update() {
System.out.println("更改了一个用户");
}
@Override
public void query() {
System.out.println("查询了一个用户");
}
}
//用这个类,自动生成代理类
public class ProxyInvocationHandler implements InvocationHandler {
//被代理的接口
private Object target;
public void setTarget(Object target) {
this.target = target;
}
//生成得到动态代理类实例
public Object getProxy(){
/*
loader: 用哪个类加载器去加载代理对象
interfaces:动态代理类需要实现的接口 (被代理接口对象所实现的所有接口)
h:动态代理方法在执行时,会调用h里面的invoke方法去执行
*/
return Proxy.newProxyInstance(this.getClass().getClassLoader(),
target.getClass().getInterfaces(),this );
}
//处理代理实例,并返回结果
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
//动态代理的本质是用反射机制来实现
log(method.getName()); //method为要执行的方法
Object result = method.invoke(target, args);
return result;
}
//附属操作
public void log(String msg){
System.out.println("执行了"+msg+"方法");
}
}
//测试类
public class Client {
public static void main(String[] args) {
//真实角色
UserServiceImpl userService = new UserServiceImpl();
ProxyInvocationHandler pih = new ProxyInvocationHandler();
//设置需要代理的对象
pih.setTarget(userService);
//动态生成代理类
UserService proxy = (UserService) pih.getProxy();
proxy.delete();
}
}
浅析Spring中AOP的实现原理:https://www.cnblogs.com/tuyang1129/p/12878549.html
3.AOP(面向切面编程)
在软件业,AOP为Aspect Oriented Programming的缩写,意为:面向切面编程,通过预编译方式和运行期动态代理实现程序功能的统一维护的一种技术。AOP是OOP的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型。利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。
AOP中的相关概念
Aspect(切面): Aspect 声明类似于 Java 中的类声明,在 Aspect 中会包含着一些 Pointcut 以及相应的 Advice。Joint point(连接点):表示在程序中明确定义的点,典型的包括方法调用,对类成员的访问以及异常处理程序块的执行等等,它自身还可以嵌套其它 joint point。Pointcut(切点):表示一组 joint point,这些 joint point 或是通过逻辑关系组合起来,或是通过通配、正则表达式等方式集中起来,它定义了相应的 Advice 将要发生的地方。Advice(增强):Advice 定义了在 Pointcut 里面定义的程序点具体要做的操作,它通过 before、after 和 around 来区别是在每个 joint point 之前、之后还是代替执行的代码。Target(目标对象):织入 Advice 的目标对象.。Weaving(织入):将 Aspect 和其他对象连接起来, 并创建 Adviced object 的过程
Advice 的类型
before advice, 在 join point 前被执行的 advice. 虽然 before advice 是在 join point 前被执行, 但是它并不能够阻止 join point 的执行, 除非发生了异常(即我们在 before advice 代码中, 不能人为地决定是否继续执行 join point 中的代码)after return advice, 在一个 join point 正常返回后执行的 adviceafter throwing advice, 当一个 join point 抛出异常后执行的 adviceafter(final) advice, 无论一个 join point 是正常退出还是发生了异常, 都会被执行的 advice.around advice, 在 join point 前和 joint point 退出后都执行的 advice. 这个是最常用的 advice.introduction,introduction可以为原有的对象增加新的属性和方法。
1.场景
//接口
public interface UserService {
public void add();
public void delete();
public void update();
public void select();
}
//实现类
public class UserServiceImpl implements UserService{
@Override
public void add() {System.out.println("增加了一个用户"); }
@Override
public void delete() {System.out.println("删除了一个用户"); }
@Override
public void update() {System.out.println("更新了一个用户"); }
@Override
public void select() {System.out.println("查询了一个用户"); }
}
2.配置
依赖
<!-- https://mvnrepository.com/artifact/org.aspectj/aspectjweaver -->
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.9.6</version>
<!--指的是直接在运行时所需要的包,而非在编译时需要的包。-->
<!--<scope>runtime</scope>-->
</dependency>
spring上下文
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:aop="http://www.springframework.org/schema/aop"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd">
<!--注册bean-->
<!--<bean id="userService" class="com.jiutian.service.UserServiceImpl"/>-->
<!--方式一-->
<!--方式二-->
<!--方式三-->
</beans>
方式一
//before
public class BeforeLog implements MethodBeforeAdvice {
//method:要执行的目标对象的方法
//args:参数
//target:目标对象
@Override
public void before(Method method, Object[] args, Object target) throws Throwable {
System.out.println(target.getClass().getName()+"的"+method.getName()+"方法被执行了");
}
}
//after
public class AfterLog implements AfterReturningAdvice {
//returnValue:返回值
@Override
public void afterReturning(Object returnValue, Method method, Object[] args, Object target) throws Throwable {
System.out.println("执行了"+method.getName()+"方法,返回结果为:"+returnValue);
}
}
//before,after
//实现方法拦截器接口
public class AroundLog implements MethodInterceptor {
@Override
public Object invoke(MethodInvocation invocation) throws Throwable {
System.out.println("++++++before advice"+"---"+invocation.getMethod().getName());
invocation.proceed();
System.out.println("++++++after advice"+"---"+invocation.getMethod().getName());
return null;
}
}
<!--方式一:使用原生Spring API接口-->
<bean id="beforelog" class="com.jiutian.log.BeforeLog"/>
<bean id="afterlog" class="com.jiutian.log.AfterLog"/>
<bean id="aroundlog" class="com.jiutian.log.AroundLog"/>
<!--配置aop: 需要导入aop约束-->
<aop:config>
<!--切入点:expression:表达式,execution(要执行的位置!* * * * *) -->
<aop:pointcut id="pointcut" expression="execution(* com.jiutian.service.UserServiceImpl.*(..))"/>
<!--执行环绕增加,aop:advisor中的通知必须实现Advice接口。-->
<aop:advisor advice-ref="beforelog" pointcut-ref="pointcut"/>
<aop:advisor advice-ref="afterlog" pointcut-ref="pointcut"/>
<!--一个顶俩-->
<!--<aop:advisor advice-ref="aroundlog" pointcut-ref="pointcut"/>-->
</aop:config>
方式二
//切面支持类
public class DiyPointCut {
//前置通知
public void before(){
System.out.println("==========方法执行前==========");
}
//后置通知
public void after(){
System.out.println("==========方法执行后==========");
}
}
<!--方式二:自定义类实现-->
<bean id="diy" class="com.jiutian.diy.DiyPointCut"/>
<aop:config>
<!--自定义切面,ref 要引用的切面支持类对象-->
<aop:aspect ref="diy">
<!--切入点-->
<aop:pointcut id="point" expression="execution(* com.jiutian.service.UserServiceImpl.*(..))"/>
<!--通知(方法)-->
<aop:before method="before" pointcut-ref="point"/>
<aop:after method="after" pointcut-ref="point"/>
</aop:aspect>
</aop:config>
方式三
@Aspect //标注这个类是一个切面类
public class AnnoPointCut {
@Before("execution(* com.jiutian.service.UserServiceImpl.*(..))")
public void before() {
System.out.println("========方法执行前(before)========");
}
@After("execution(* com.jiutian.service.UserServiceImpl.*(..))")
public void after() {
System.out.println("========方法执行后(after)========");
}
@AfterReturning("execution(* com.jiutian.service.UserServiceImpl.*(..))")
public void afterR() {
System.out.println("========方法执行后(afterReturning)========");
}
//在环绕增强中,可以给定一个参数,代表要获取处理切入的点
@Around("execution(* com.jiutian.service.UserServiceImpl.*(..))")
public void around(ProceedingJoinPoint pj) throws Throwable {
System.out.println("环绕前");
//Signature signature = pj.getSignature(); //获得签名
//System.out.println("signature:" + signature);
Object proceed = pj.proceed(); //执行方法
System.out.println("环绕后");
System.out.println(proceed);
}
}
<!--方式三:使用注解-->
<bean id="annoPointCut" class="com.jiutian.diy.AnnoPointCut"/>
<!--开启注解支持 JDK(默认 proxy-target-class="false") cglib(proxy-target-class="true")-->
<aop:aspectj-autoproxy proxy-target-class="true"/>
2.测试
//测试
public class MyTest {
public static void main(String[] args) {
ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
//动态代理的是接口
UserService userService = (UserService) context.getBean("userService");
userService.add();
}
}
4.整合Mybatis
1.所需依赖
<dependencies>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.13</version>
</dependency>
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>8.0.21</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-jdbc</artifactId>
<version>5.3.9</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-webmvc</artifactId>
<version>5.3.9</version>
</dependency>
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.9.6</version>
</dependency>
<dependency>
<groupId>org.mybatis</groupId>
<artifactId>mybatis</artifactId>
<version>3.5.6</version>
</dependency>
<!-- https://mvnrepository.com/artifact/org.mybatis/mybatis-spring -->
<dependency>
<groupId>org.mybatis</groupId>
<artifactId>mybatis-spring</artifactId>
<version>2.0.6</version>
</dependency>
<dependency>
<groupId>org.projectlombok</groupId>
<artifactId>lombok</artifactId>
<version>1.18.20</version>
</dependency>
</dependencies>
<!--在build中配置resources,来防止我们资源导出失败的问题-->
<build>
<resources>
<resource>
<directory>src/main/resources</directory>
<includes>
<include>**/*.properties</include>
<include>**/*.xml</include>
</includes>
<filtering>true</filtering>
</resource>
<resource>
<directory>src/main/java</directory>
<includes>
<include>**/*.properties</include>
<include>**/*.xml</include>
</includes>
<filtering>true</filtering>
</resource>
</resources>
</build>
2.配置文件
mybatis-config.xml
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE configuration
PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-config.dtd">
<!--核心配置文件-->
<configuration>
<!--<settings>-->
<!-- <setting name="" value=""/>-->
<!--</settings>-->
<typeAliases>
<package name="com.jiutian.pojo"/>
</typeAliases>
</configuration>
spring-dao.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:aop="http://www.springframework.org/schema/aop"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd">
<!--数据源,使用Spring的数据源替代Mybatis的配置 c3p0 dbcp druid-->
<!--这里使用Spring提供的JDBC-->
<bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
<property name="driverClassName" value="com.mysql.cj.jdbc.Driver"/>
<property name="url" value="jdbc:mysql://localhost:3306/mybatis?useSSL=true&useUnicode=true&characterEncoding=UTF-8&serverTimezone=GMT%2B8"/>
<property name="username" value="root"/>
<property name="password" value="123456"/>
</bean>
<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
<property name="dataSource" ref="dataSource" />
<!--绑定mybatis配置文件,可以不用-->
<property name="configLocation" value="classpath:mybatis-config.xml"/>
<!--绑定mapper文件-->
<property name="mapperLocations" value="classpath:com/jiutian/mapper/*.xml"/>
</bean>
<bean id="sqlSession" class="org.mybatis.spring.SqlSessionTemplate">
<constructor-arg index="0" ref="sqlSessionFactory"/>
</bean>
</beans>
3.applicationContext.xml
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:aop="http://www.springframework.org/schema/aop"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd">
<import resource="spring-dao.xml"/>
<bean id="userMapper" class="com.jiutian.mapper.UserMapperImpl">
<property name="sqlSession" ref="sqlSession"/>
</bean>
<bean id="userMapper2" class="com.jiutian.mapper.UserMapperImpl2">
<property name="sqlSessionFactory" ref="sqlSessionFactory"/>
</bean>
</beans>
3.Mapper
接口
public interface UserMapper {
List<User> selectUser();
}
mapper.xml
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Config 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.jiutian.mapper.UserMapper">
<select id="selectUser" resultType="User">
select * from mybatis.user
</select>
</mapper>
实现类 (用于获取sqlSession Bean)
方式一(注入SqlSessionTemplate)
public class UserMapperImpl implements UserMapper{
private SqlSessionTemplate sqlSession;
public void setSqlSession(SqlSessionTemplate sqlSession) {
this.sqlSession = sqlSession;
}
@Override
public List<User> selectUser() {
UserMapper mapper = sqlSession.getMapper(UserMapper.class);
return mapper.selectUser();
}
}
方式二(注入SqlSessionFactory)
public class UserMapperImpl2 extends SqlSessionDaoSupport implements UserMapper{
@Override
public List<User> selectUser() {
return getSqlSession().getMapper(UserMapper.class).selectUser();
}
}
4.测试
@Test
public void test(){
ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
UserMapper userMapper = context.getBean("userMapper2", UserMapper.class);
List<User> userList = userMapper.selectUser();
for (User user : userList) {
System.out.println(user);
}
}
5.事务
1.要点
-
把一组业务当作一个业务来做,要么都成功,要么都失败!
-
确保一致性和完整性。
事务的ACID原则:
-
原子性: 一个事务是一个不可分割的工作单位,其中的操作要么都做,要么都不做。
-
一致性:事务操作成功后,数据库所处的状态和他的业务规则是一致的,即数据不会被破坏。
-
隔离性:事务内部的操作与其他事务是隔离的,并发执行的各个事务之间不能互相干扰。
-
持久性 :事务一旦提交,它对数据库的改变就应该是永久性的。接下来的其他操作或故障不应该对其有任何影响。
spring中的事务管理
- 声明式事务:AOP
- 编程式事务:需要在代码中进行事务管理
2.声明式事务
mapper接口
public interface UserMapper {
public List<User> selectUser();
public int addUser(User user);
public int deleteUser(@Param("id") int id);
public int updateUser(User user);
}
实现类
public class UserMapperImpl extends SqlSessionDaoSupport implements UserMapper{
@Override
public List<User> selectUser() {
User user = new User(5, "qqq", "123");
UserMapper mapper = getSqlSession().getMapper(UserMapper.class);
mapper.addUser(user);
mapper.deleteUser(4);
return mapper.selectUser();
}
@Override
public int addUser(User user) {
return getSqlSession().getMapper(UserMapper.class).addUser(user);
}
@Override
public int deleteUser(int id) {
return getSqlSession().getMapper(UserMapper.class).deleteUser(id);
}
@Override
public int updateUser(User user) {
return getSqlSession().getMapper(UserMapper.class).updateUser(user);
}
}
关键配置
<!--配置声明式事务-->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"/>
</bean>
<!--结合Aop实现事务的织入-->
<!--配置事务的通知-->
<tx:advice id="txAdvice" transaction-manager="transactionManager">
<!--给哪些方法配置事务-->
<!--配置事务的传播特性: propagation:默认REQUIRED-->
<tx:attributes>
<tx:method name="add" propagation="REQUIRED"/>
<tx:method name="delete"/>
<tx:method name="update"/>
<tx:method name="query" read-only="true"/>
<tx:method name="*"/>
</tx:attributes>
</tx:advice>
<!--配置事务切入-->
<aop:config>
<aop:pointcut id="txPointCut" expression="execution(* com.jiutian.mapper.*.*(..))"/>
<aop:advisor advice-ref="txAdvice" pointcut-ref="txPointCut"/>
</aop:config>
测试
@Test
public void test(){
ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
UserMapper userMapper = context.getBean("userMapper", UserMapper.class);
List<User> userList = userMapper.selectUser();
for (User user : userList) {
System.out.println(user);
}
}