Github代码
【狂神说JAVA】Spring5笔记(所有代码均经过测试)
从11开始是因为后面学习改狂神说,前面是尚硅谷
11、使用Spring实现Aop
SpringAOP中,通过Advice定义横切逻辑,Spring中支持5种类型的Advice:
通知类型 | 连接点 | 实现接口 |
---|---|---|
前置通知 | 方法前 | org.springframework.aop.MethodBeforeAdvice |
后置通知 | 方法后 | org.springframework.aop.AfterReturningAdvice |
环绕通知 | 方法前后 | org.aopalliance.intercept.MethodInterceptor |
异常抛出通知 | 方法抛出异常 | org.springframework.aop.ThrowsAdvice |
引介通知 | 类中增加新的方法属性 | org.springframework.aop.IntroductionInterceptor |
即 Aop 在 不改变原有代码的情况下 , 去增加新的功能 .
11.3 使用Spring实现Aop
【重点】使用AOP织入,需要导入一个依赖包!
<!-- https://mvnrepository.com/artifact/org.aspectj/aspectjweaver --><dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.9.4</version>
</dependency>
11.3.1 方式一:使用Spring的API接口
【主要SpringAPI接口实现】
首先编写我们的业务接口和实现类
package com.kuang.service;
public interface UserService {
public String add();
public void delete();
public void update();
public void select();
}
package com.kuang.service;
public class UserServiceImpl implements UserService{
@Override
public String add() {
System.out.println("增加用户");
return "sss";
}
@Override
public void delete() {
System.out.println("删除用户");
}
@Override
public void update() {
System.out.println("更新用户");
}
@Override
public void select() {
System.out.println("查询用户");
}
}
- 写我们的增强类 , 我们编写两个 , 一个前置增强 一个后置增强
package com.kuang.log;
import org.springframework.aop.MethodBeforeAdvice;
import java.lang.reflect.Method;
public class log implements MethodBeforeAdvice {
//method : 要执行的目标对象的方法
//objects : 被调用的方法的参数
//Object : 目标对象
@Override
public void before(Method method, Object[] objects, Object o) throws Throwable {
System.out.println( o.getClass().getName() + "的" + method.getName() + "方法被执行了");
}
}
package com.kuang.log;
import org.springframework.aop.AfterReturningAdvice;
import java.lang.reflect.Method;
public class AfterLog implements AfterReturningAdvice {
//returnValue 返回值
//method被调用的方法
//args 被调用的方法的对象的参数
//target 被调用的目标对象
@Override
public void afterReturning(Object returnValue, Method method, Object[] args, Object target) throws Throwable {
System.out.println("执行了" + target.getClass().getName()
+"的"+method.getName()+"方法,"
+"返回值:"+returnValue);
}
}
- 在spring的文件中注册 , 并实现aop切入实现 , 注意导入约束 .
<?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.kuang.service.UserServiceImpl"/>
<bean id="log" class="com.kuang.log.log"/>
<bean id="afterLog" class="com.kuang.log.AfterLog"/>
<!--aop的配置-->
<aop:config>
<!--切入点 expression:表达式匹配要执行的方法-->
<aop:pointcut id="pointcut" expression="execution(* com.kuang.service.UserServiceImpl.*(..))"/>
<!--执行环绕; advice-ref执行方法 . pointcut-ref切入点-->
<aop:advisor advice-ref="log" pointcut-ref="pointcut"/>
<aop:advisor advice-ref="afterLog" pointcut-ref="pointcut"/>
</aop:config>
</beans>
- 测试
package com.kuang.service;
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class MyTest {
@Test
public void test(){
ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
UserService userService = (UserService) context.getBean("userService");
userService.add();
}
}
com.kuang.service.MyTest,test
com.kuang.service.UserServiceImpl的add方法被执行了
增加用户
执行了com.kuang.service.UserServiceImpl的add方法,返回值:sss
进程已结束,退出代码为 0
Aop的重要性 : 很重要 . 一定要理解其中的思路 , 主要是思想的理解这一块 .
**Spring的Aop就是将公共的业务 (日志 , 安全等) 和领域业务结合起来 , 当执行领域业务时 , 将会把公共业务加进来 . 实现公共业务的重复利用 . 领域业务更纯粹 , 程序猿专注领域业务 , 其本质还是动态代理 . **
11.3.2 方式二:自定义来实现【主要是切面定义】
目标业务类不变依旧是userServiceImpl
- 写我们自己的一个切入类
package com.kuang.diy;
public class DiyPointCut {
public void before() {
System.out.println("==========方法执行前============");
}
public void after() {
System.out.println("==========方法执行后============");
}
}
- 去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.kuang.service.UserServiceImpl"/>
<!--注册bean-->
<bean id="diy" class="com.kuang.diy.DiyPointCut"/>
<!--aop的配置-->
<aop:config>
<!--自定义切面,ref要引用的类-->
<aop:aspect ref="diy">
<!--切入点-->
<aop:pointcut id="point" expression="execution(* com.kuang.service.UserServiceImpl.*(..))"/>
<aop:before method="before" pointcut-ref="point"/>
<aop:after method="after" pointcut-ref="point"/>
<!--通知-->
</aop:aspect>
</aop:config>
</beans>
@Test
public void test2(){
ApplicationContext context = new ClassPathXmlApplicationContext("beans2.xml");
UserService userService = (UserService) context.getBean("userService");
userService.add();
}
==========方法执行前============
增加用户
==========方法执行后============
进程已结束,退出代码为 0
11.3.3 方式三:使用注解实现!
- 编写一个注解实现的增强类
package com.kuang.diy;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.After;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Before;
//方式三:使用注解的方式实现AOP
@Aspect//标注这个类是一个切面
public class AnnotationPointCut {
@Before("execution(* com.kuang.service.UserServiceImpl.*(..))")
public void before() {
System.out.println("==========方法执行前============");
}
@After("execution(* com.kuang.service.UserServiceImpl.*(..))")
public void after() {
System.out.println("==========方法执行后============");
}
//在环绕增强中,我们可以给定一个参数,代表我们要获取处理切入的点
@Around("execution(* com.kuang.service.UserServiceImpl.*(..))")
public void around(ProceedingJoinPoint jp) throws Throwable {
System.out.println("==========环绕前============");
//获取签名
//Signature signature = jp.getSignature();
//System.out.println("signature:" + signature);
//执行方法
Object proceed = jp.proceed();
System.out.println("==========环绕后============");
}
}
- 在Spring配置文件中,注册bean,并增加支持注解的配置
<?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.kuang.service.UserServiceImpl"/>
<!--方式三:注解-->
<bean id="annotationPointCut" class="com.kuang.diy.AnnotationPointCut"/>
<!--开启注解支持: JDK(默认proxy-target-class="false") cglib("true“)-->
<aop:aspectj-autoproxy proxy-target-class="false"/>
</beans>
- 测试
@Test
public void test3(){
ApplicationContext context = new ClassPathXmlApplicationContext("beans3.xml");
UserService userService = (UserService) context.getBean("userService");
userService.add();
}
==========环绕前============
==========方法执行前============
增加用户
==========方法执行后============
==========环绕后============
进程已结束,退出代码为 0
12、整合Mybatis
12.1 导入相关jar包
- junit
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.12</version>
</dependency>
- mybatis
<!--mybatis-->
<dependency>
<groupId>org.mybatis</groupId>
<artifactId>mybatis</artifactId>
<version>3.5.4</version>
</dependency>
- mysql数据库:mysql-connector-java
<!--mysqlq驱动-->
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>8.0.22</version>
</dependency>
- spring相关的:spring-webmvc
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-webmvc</artifactId>
<version>5.1.10.RELEASE</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-jdbc</artifactId>
<version>5.1.10.RELEASE</version>
</dependency>
- aspectJ AOP 织入器
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.9.4</version>
</dependency>
- mybatis-spring整合包【重点】
<dependency>
<groupId>org.mybatis</groupId>
<artifactId>mybatis-spring</artifactId>
<version>2.0.2</version>
</dependecy>
- 配置Maven静态资源过滤问题!【约定大于配置】
<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>
12.2 编写mybatis项目
12.3 Mybatis-Spring
- 什么是 MyBatis-Spring?
MyBatis-Spring 会帮助你将 MyBatis 代码无缝地整合到 Spring 中。
如果使用 Maven 作为构建工具,仅需要在 pom.xml 中加入以下代码即可:
<dependency>
<groupId>org.mybatis</groupId>
<artifactId>mybatis-spring</artifactId>
<version>2.0.2</version>
</dependency>
12.3.1 整合方式一
- 引入配置文件
<?xml version="1.0" encoding="UTF-8" ?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd">
</beans>
- 编写数据源配置
<bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
<property name="driverClassName" value="com.mysql.jdbc.Driver"/>
<property name="url" value="jdbc:mysql://localhost:3306/mybatis?userSSL=true&useUnicode=true&characterEncoding=UTF-8&serverTimezone=UTC"/>
<property name="username" value="mybt"/>
<property name="password" value="123456"/>
</bean>
- 配置sqlSessionFactory
<!--配置SqlSessionFactory-->
<bean id="sqlSessionFactory" class="org.mybatis.spring.SqlSessionFactoryBean">
<property name="dataSource" ref="dataSource"/>
<!--关联Mybatis-->
<property name="configLocation" value="classpath:mybatis-config.xml"/>
<property name="mapperLocations" value="classpath:com/kuang/mapper/UserMapper.xml"/>
</bean>
- sqlSessionTemplate
<!--注册sqlSessionTemplate , 关联sqlSessionFactory-->
<bean id="sqlSession" class="org.mybatis.spring.SqlSessionTemplate">
<!--利用构造器注入,没有set注入,只能使用构造器注入-->
<constructor-arg index="0" ref="sqlSessionFactory"/>
</bean>
- 需要给接口加实现类【新加的】
package com.kuang.mapper;
import com.kuang.pojo.User;
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import java.util.List;
public class TestDao {
@Test
public void getUserList() {
ApplicationContext context = new ClassPathXmlApplicationContext("ApplicationContext.xml");
UserMapper userDao = context.getBean("userMapper",UserMapper.class);
List<User> allUser = userDao.getUserList();
for (User user : allUser) {
System.out.println(user);
}
}
}
- 将自己写的实现类,注入到Spring中
<bean id="userDao" class="Com.Sun.Dao.UserDaoImpl">
<property name="sqlSession" ref="sqlSession"/>
</bean>
- 测试使用即可
package com.kuang.mapper;
import com.kuang.pojo.User;
import com.kuang.utils.MybatisUtils;
import org.apache.ibatis.session.SqlSession;
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import java.util.List;
public class TestDao {
@Test
public void getUserList2() {
ApplicationContext context = new ClassPathXmlApplicationContext("spring-dao.xml");
UserMapper userDao = (UserMapper) context.getBean("userDao");
List<User> allUser = userDao.getUserList();
for (User user : allUser) {
System.out.println(user);
}
}
}
User(id=1, name=张三, pwd=123)
User(id=2, name=张三, pwd=123)
User(id=3, name=王五, pwd=789)
User(id=4, name=哈哈, pwd=454)
User(id=7, name=热热热, pwd=7787)
User(id=5, name=王发的, pwd=457)
User(id=8, name=张强, pwd=5754)
User(id=9, name=大款, pwd=4574)
User(id=10, name=张强, pwd=5754)
- 为了给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核心配置文件-->
<configuration>
<typeAliases>
<package name="com.kuang.pojo"/>
</typeAliases>
</configuration>
12.3.2 整合方式二(推荐)
- 将我们上面写的UserDaoImpl修改一下
package com.kuang.mapper;
import com.kuang.pojo.User;
import org.mybatis.spring.support.SqlSessionDaoSupport;
import java.util.List;
public class UserDaoImpl2 extends SqlSessionDaoSupport implements UserMapper {
@Override
public List<User> getUserList() {
UserMapper mapper = getSqlSession().getMapper(UserMapper.class);
List<User> allUser = mapper.getUserList();
return allUser;
}
}
- 修改bean的配置
<bean id="userMapper2" class="com.kuang.mapper.UserDaoImpl2">
<property name="sqlSessionFactory" ref="sqlSessionFactory"/>
</bean>
- 测试
@Test
public void getUserList2() {
ApplicationContext context = new ClassPathXmlApplicationContext("ApplicationContext.xml");
UserMapper userDao = context.getBean("userMapper2",UserMapper.class);
List<User> allUser = userDao.getUserList();
for (User user : allUser) {
System.out.println(user);
}
}
总结 : 整合到spring以后可以完全不要mybatis的配置文件,除了这些方式可以实现整合之外,我们还可以使用注解来实现,这个等我们后面学习SpringBoot的时候还会测试整合!
13、声明式事务
13.1 回顾事务
- 把一组业务当成一个业务来做;要么都成功,要么都失败!
- 事务在项目开发中,十分重要,涉及到数据的一致性问题,不能马虎
- 确保完整性和一致性
事务的ACID原则:
- 原子性(atomicity)
事务是原子性操作,由一系列动作组成,事务的原子性确保动作要么全部完成,要么完全不起作用 - 一致性(consistency)
一旦所有事务动作完成,事务就要被提交。数据和资源处于一种满足业务规则的一致性状态中 - 隔离性(isolation)
多个业务可能操作同一个资源,防止数据损坏 - 持久性(durability)
事务一旦提交,无论系统发生什么问题,结果都不会再被影响,被持久化的写到存储器中!
13.2 案例
新建用户再删除
public int addUser(User user);
public int deleteUserById(int id);
- 正确的addUser,错误的deleteUserById
<insert id="addUser" parameterType="User">
insert user (name, pwd) value (#{name}, #{pwd});
</insert>
<delete id="deleteUserById">
deletes user where id = #{id}
</delete>
UserDaoImpl
package com.kuang.mapper;
import com.kuang.pojo.User;
import org.mybatis.spring.support.SqlSessionDaoSupport;
import java.util.List;
public class UserDaoImpl extends SqlSessionDaoSupport implements UserMapper {
@Override
public List<User> getUserList() {
UserMapper mapper = getSqlSession().getMapper(UserMapper.class);
User user = new User(0,"小王","7458");
int id = mapper.addUser(user);
mapper.deleteUserById(id);
List<User> allUser = mapper.getUserList();
return allUser;
}
public int addUser(User user){
return getSqlSession().getMapper(UserMapper.class).addUser(user);
}
public int deleteUserById(int id){
return getSqlSession().getMapper(UserMapper.class).deleteUserById(id);
}
}
- 测试
@Test
public void getUserList() {
ApplicationContext context = new ClassPathXmlApplicationContext("ApplicationContext.xml");
UserMapper userDao = context.getBean("userMapper",UserMapper.class);
List<User> allUser = userDao.getUserList();
for (User user : allUser) {
System.out.println(user);
}
}
- 中断但插入成功,却未删除,不符事务
13.3 Spring中的事务管理
-
声明式事务:AOP(推荐)
- 一般情况下比编程式事务好用。
- 将事务管理代码从业务方法中分离出来,以声明的方式来实现事务管理。
- 将事务管理作为横切关注点,通过aop方法模块化。Spring中通过Spring AOP框架支持声明式事务管理
-
编程式事务:需要在代码中,进行事务的管理
- 将事务管理代码嵌到业务方法中来控制事务的提交和回滚
- 缺点:必须在每个事务操作业务逻辑中包含额外的事务管理代码
-
使用Spring管理事务,注意头文件的约束导入 :
tx
(事务)
xmlns:tx="http://www.springframework.org/schema/tx"
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx.xsd">
-
事务管理器
- 无论使用Spring的哪种事务管理策略(编程式或者声明式)事务管理器都是必须的。
- 就是 Spring的核心事务管理抽象,管理封装了一组独立于技术的方法。
-
JDBC事务
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource" />
</bean>
- 配置事务的通知
<!--配置事务通知-->
<tx:advice id="txAdvice" transaction-manager="transactionManager">
<tx:attributes>
<!--配置哪些方法使用什么样的事务,配置事务的传播特性-->
<tx:method name="add" propagation="REQUIRED"/>
<tx:method name="delete" propagation="REQUIRED"/>
<tx:method name="update" propagation="REQUIRED"/>
<tx:method name="search*" propagation="REQUIRED"/>
<tx:method name="get" read-only="true"/>
<tx:method name="*" propagation="REQUIRED"/>
</tx:attributes>
</tx:advice>
- 配置事务的切入
<aop:config>
<aop:pointcut id="txPointCut" expression="execution(* com.kuang.mapper.*.*(..))"/>
<aop:advisor advice-ref="txAdvice" pointcut-ref="txPointCut"/>
</aop:config>
版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。
文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/92665.html