Spring5需要学习的内容:
内容不多大约5天可以学完
1
、
Spring
是轻量级的开源的
JavaEE
框架
2
、
Spring
可以
解决企业应用开发的复杂性
3
、
Spring
有两个核心部分:
IOC
和
Aop
(
1
)
IOC
:控制反转,把创建对象过程交给
Spring
进行管理
(2)
Aop
:面向切面,不修改源代码进行功能增强
4
、
Spring
特点
(
1
)方便解耦,简化开发
(2)
Aop
编程支持
(3)方便程序测试
(4)方便和其他框架进行整合
(5)方便进行事务操作
(6)降低
API
开发难度
Spring下载地址
ps://repo.spring.io/release/org/springframework/spring/
1、入门案例
我们需要导入Spring对应的jar包也可以直接使用maven导入依赖 推荐使用maven因为maven导入依赖方便
找到mavne库
http://mvnrepoitory.com在搜索框输入spring就行然后赋值代码 放到maven库的配置文件中即可 一定要把代码放入到dependencies中
<dependencies>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring</artifactId>
<version>5.3.17</version>
<type>pom</type>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>RELEASE</version>
<scope>compile</scope>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>RELEASE</version>
<scope>compile</scope>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>5.3.17</version>
</dependency>
<!-- https://mvnrepository.com/artifact/org.springframework/spring-web -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-web</artifactId>
<version>5.3.17</version>
</dependency>
<!-- https://mvnrepository.com/artifact/org.springframework/spring-aop -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-aop</artifactId>
<version>5.3.17</version>
</dependency>
<!-- https://mvnrepository.com/artifact/org.springframework/spring-context -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>5.3.17</version>
</dependency>
<!-- https://mvnrepository.com/artifact/org.springframework/spring-aspects -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-aspects</artifactId>
<version>5.3.18</version>
</dependency>
<!-- https://mvnrepository.com/artifact/mysql/mysql-connector-java -->
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>5.1.8</version>
</dependency>
<!-- https://mvnrepository.com/artifact/com.alibaba/druid -->
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>druid</artifactId>
<version>1.2.8</version>
</dependency>
<!-- https://mvnrepository.com/artifact/org.springframework/spring-jdbc -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-jdbc</artifactId>
<version>5.3.18</version>
</dependency>
<!-- https://mvnrepository.com/artifact/org.springframework/spring-tx -->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-tx</artifactId>
<version>5.3.18</version>
</dependency>
<!-- https://mvnrepository.com/artifact/org.apache.servicemix.bundles/org.apache.servicemix.bundles.spring-tx -->
<dependency>
<groupId>org.apache.servicemix.bundles</groupId>
<artifactId>org.apache.servicemix.bundles.spring-tx</artifactId>
<version>5.3.16_1</version>
</dependency>
</dependencies>
创建xml文件
创建一个book类
public class book {
public void add(){
System.out.println("book add....");
}
}
写入配置文件
<?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">
<!-- 配置User对象创建-->
<bean id="user" class="com.atgjk.service.book"></bean>
</beans>
id就是自己起的名字 class后面加的是全路径
写入测试类
public class textBook {
@Test
public void book(){
//1、加载配置文件
// ClassPathXmlApplicationContext 这个是读取写在src目录下边的 括号里填写的内容是文件的名字
// FileSystemXmlApplicationContext 这个是读取写在其他目录下 里边填写的内容是全路径
ApplicationContext context =new FileSystemXmlApplicationContext("E:\\Spring5\\textSpring\\src\\main\\java\\com\\atgjk\\xml\\bean.xml");
book user = context.getBean("user", book.class);
user.add();
}
}
这样就能运行出来 book类里边的 book add ...
平时如果想调用add方法只能通过new的方式来调用 现在可以直接通过配置文件的方式来调用这样以后想改变的话就方便了 只需要改变配置文件即可 不需要改动源代码
1
、什么是
IOC
(
1
)控制反转,把对象创建和对象之间的调用过程,交给
Spring
进行管理
(2)使用
IOC
目的:为了耦合度降低
(3)做入门案例就是
IOC
实现
2
、
IOC
底层原理
(
1
)
xml
解析、工厂模式、反射
IOC
思想基于
IOC
容器完成,
IOC
容器底层就是对象工厂
2
、
Spring
提供
IOC
容器实现两种方式:(两个接口)
(
1
)
BeanFactory
:
IOC
容器基本实现,是
Spring
内部的使用接口,不提供开发人员进行使用
*
加载配置文件时候不会创建对象,在获取对象(使用)才去创建对象
(2)
ApplicationContext
:
BeanFactory
接口的子接口,提供更多更强大的功能,一般由开发人
员进行使用
*
加载配置文件时候就会把在配置文件对象进行创建
1
、什么是
Bean
管理
(0)
Bean
管理指的是两个操作
(
1
)
Spring
创建对象
(2)
Spirng
注入属性
2
、
Bean
管理操作有两种方式
(
1
)基于
xml
配置文件方式实现
(2)基于注解方式实现
1
、基于
xml
方式创建对象
(
1
)在
spring
配置文件中,使用
bean
标签,标签里面添加对应属性,就可以实现对象创建
(2)在
bean
标签有很多属性,介绍常用的属性
* id
属性:唯一标识
* class
属性:类全路径(包类路径)
(3)创建对象时候,默认也是执行无参数构造方法完成对象创建
入门案例就是用的xml创建对象
2
、基于
xml
方式注入属性
(
1
)
DI
:依赖注入,就是注入属性
3
、第一种注入方式:使用
set
方法进行注入
(
1
)创建类,定义属性和对应的
set
方法
/**
* 演示使用 set 方法进行注入属性
*/
public class Book {
//创建属性
private String bname;
private String bauthor;
//创建属性对应的 set 方法
public void setBname(String bname) {
this.bname = bname;
}
public void setBauthor(String bauthor) {
this.bauthor = bauthor;
} }(2)在 spring 配置文件配置对象创建,配置属性注入
<!--2 set 方法注入属性--> <bean id="book" class="com.atguigu.spring5.Book">
<!--使用 property 完成属性注入
name:类里面属性名称
value:向属性注入的值
-->
<property name="bname" value="易筋经"></property>
<property name="bauthor" value="达摩老祖"></property>
</bean>
4
、第二种注入方式:使用有参数构造进行注入
(
1
)创建类,定义属性,创建属性对应有参数构造方法
/**
* 使用有参数构造注入
*/
public class
Orders {
//属性
private
String
oname
;
private
String
address
;
//有参数构造
public
Orders(String oname,String address) {
this
.
oname
= oname;
this
.
address
= address;
}
}
(
2
)在
spring
配置文件中进行配置
<!--3 有参数构造注入属性-->
<
bean
id=
"orders"
class=
"com.atguigu.spring5.Orders"
>
<
constructor-arg
name=
"oname"
value=
"电脑"
></
constructor-arg
>
<
constructor-arg
name=
"address"
value=
"China"
></
constructor-arg
>
</
bean
>
IOC
操作
Bean
管理(
xml
注入其他类型属性)
1
、字面量
(
1
)
null
值
<!--null 值-->
<
property
name=
"address"
>
<
null
/>
</
property
>
(
2
)属性值包含特殊符号
<!--属性值包含特殊符号
1 把<>进行转义 < >
2 把带特殊符号内容写到 CDATA
-->
<
property
name=
"address"
>
<
value
><![CDATA[<<南京>>]]></
value
>
</
property
bean分为内部bean和外部bean 上边的入门案例就是外部bean 内部bean是在bean里边嵌套bean
注入属性-级联赋值
(
1)第一种写法
<!--级联赋值-->
<
bean
id=
"emp"
class=
"com.atguigu.spring5.bean.Emp"
>
<!--设置两个普通属性-->
<
property
name=
"ename"
value=
"lucy"
></
property
>
<
property
name=
"gender"
value=
"女"
></
property
>
<!--级联赋值-->
<
property
name=
"dept"
ref=
"dept"
></
property
>
</
bean
>
<
bean
id=
"dept"
class=
"com.atguigu.spring5.bean.Dept"
>
<
property
name=
"dname"
value=
"财务部"
></
property
>
</
bean
>
就是直接调用另一个bean
<!--1 集合类型属性注入--> <bean id="stu" class="com.atguigu.spring5.collectiontype.Stu">
<!--数组类型属性注入-->
<property name="courses">
<array>
<value>java 课程</value>
<value>数据库课程</value>
</array>
</property>
<!--list 类型属性注入-->
<property name="list">
<list>
<value>张三</value>
<value>小三</value>
</list>
</property>
<!--map 类型属性注入-->
<property name="maps">
<map>
<entry key="JAVA" value="java"></entry>
<entry key="PHP" value="php"></entry>
</map>
</property>
<!--set 类型属性注入-->
<property name="sets">
<set>
<value>MySQL</value>
<value>Redis</value>
</set>
</property>
</bean>
把集合注入部分提取出来
(1)在 spring 配置文件中引入名称空间 util
<?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:p="http://www.springframework.org/schema/p"
xmlns:util="http://www.springframework.org/schema/util"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/util
http://www.springframework.org/schema/util/spring-util.xsd"> (2)使用 util 标签完成 list 集合注入提取
<!--1 提取 list 集合类型属性注入--> <util:list id="bookList">
<value>易筋经</value>
<value>九阴真经</value>
<value>九阳神功</value>
</util:list>
<!--2 提取 list 集合类型属性注入使用--> <bean id="book" class="com.atguigu.spring5.collectiontype.Book">
<property name="list" ref="bookList"></property>
</bean>
1
、
Spring
有两种类型
bean
,一种普通
bean
,另外一种工厂
bean
(
FactoryBean
)
2
、普通
bean
:在配置文件中定义
bean
类型就是返回类型
3
、工厂
bean
:在配置文件定义
bean
类型可以和返回类型不一样
第一步 创建类,让这个类作为工厂
bean
,实现接口
FactoryBean
第二步 实现接口里面的方法,在实现的方法中定义返回的
bean
类型
4、如何设置单实例还是多实例
(
1
)在
spring
配置文件
bean
标签里面有属性(
scope
)用于设置单实例还是多实例
(2)
scope
属性值
第一个值
默认值,
singleton
,表示是单实例对象
第二个值
prototype
,表示是多实例对象
(3)
singleton
和
prototype
区别
第一
singleton
单实例,
prototype
多实例
第二
设置
scope
值是
singleton
时候,加载
spring
配置文件时候就会创建单实例对象
设置
scope
值是
prototype
时候,不是在加载
spring
配置文件时候创建 对象,在调用
getBean
方法时候创建多实例对象
IOC
操作
Bean
管理
(
外部属性文件
)
1
、直接配置数据库信息
(
1
)配置德鲁伊连接池
(2)引入德鲁伊连接池依赖
jar
包
<!--直接配置连接池-->
<
bean
id
="dataSource"
class
="com.alibaba.druid.pool.DruidDataSource"
>
<
property
name
="driverClassName"
value
="com.mysql.jdbc.Driver"
></
property
>
<
property
name
="url"
value
="jdbc:mysql://localhost:3306/userDb"
></
property
>
<
property
name
="username"
value
="root"
></
property
>
<
property
name
="password"
value
="root"
></
property
>
</
bean
>
2
、引入外部属性文件配置数据库连接池
(
1
)创建外部属性文件,
properties
格式文件,写数据库信息
(2)把外部 properties 属性文件引入到 spring 配置文件中
* 引入 context 名称空间
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:p="http://www.springframework.org/schema/p"
xmlns:util="http://www.springframework.org/schema/util"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/util
http://www.springframework.org/schema/util/spring-util.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd"> ⚫ 在 spring 配置文件使用标签引入外部属性文件
<!--引入外部属性文件--> <context:property-placeholder location="classpath:jdbc.properties"/>
<!--配置连接池--> <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource">
<property name="driverClassName" value="${prop.driverClass}"></property>
<property name="url" value="${prop.url}"></property>
<property name="username" value="${prop.userName}"></property>
<property name="password" value="${prop.password}"></property>
</bean>
IOC
操作
Bean
管理
(
基于注解方式
)
1
、什么是注解
(
1
)注解是代码特殊标记,格式:
@
注解名称
(
属性名称
=
属性值
,
属性名称
=
属性值
..)
(2)使用注解,注解作用在类上面,方法上面,属性上面
(3)使用注解目的:简化
xml
配置
2
、
Spring
针对
Bean
管理中创建对象提供注解
(
1
)
@Component
(2)
@Service
(3)
@Controller
(4)
@Repository
*
上面四个注解功能是一样的,都可以用来创建
bean
实例
3
、基于注解方式实现对象创建
第一步 引入依赖
第二步 开启组件扫描
<!--开启组件扫描
1 如果扫描多个包,多个包使用逗号隔开
2 扫描包上层目录
-->
<
context
:component-scan
base-package=
"com.atguigu"
></
context
:component-scan
>
第三步 创建类,在类上面添加创建对象注解
//在注解里面 value 属性值可以省略不写,
//默认值是类名称,首字母小写
//UserService -- userService
@Component
(value =
"userService"
)
//<bean id="userService" class=".."/>
public class
UserService {
public void
add() {
System.
out
.println(
"service add......."
);
}
}
4
、开启组件扫描细节配置
<!--示例 1
use-default-filters="false" 表示现在不使用默认 filter,自己配置 filter
context:include-filter ,设置扫描哪些内容
--><context:component-scan base-package="com.atguigu" use-defaultfilters="false">
<context:include-filter type="annotation"
expression="org.springframework.stereotype.Controller"/>
</context:component-scan>
<!--示例 2
下面配置扫描包所有内容
context:exclude-filter: 设置哪些内容不进行扫描
--><context:component-scan base-package="com.atguigu">
<context:exclude-filter type="annotation"
expression="org.springframework.stereotype.Controller"/>
</context:component-scan>
5
、基于注解方式实现属性注入
(
1
)
@Autowired
:根据属性类型进行自动装配
第一步 把
service
和
dao
对象创建,在
service
和
dao
类添加创建对象注解
第二步 在
service
注入
dao
对象,在
service
类添加
dao
类型属性,在属性上面使用注解
@Service
public class
UserService {
//定义 dao 类型属性
//不需要添加 set 方法
//添加注入属性注解
@Autowired
private
UserDao
userDao
;
public void
add() {
System.
out
.println(
"service add......."
);
userDao
.add();
}
}
(
2
)
@Qualifier
:根据名称进行注入
这个
@Qualifier
注解的使用,和上面
@Autowired
一起使用
//定义 dao 类型属性
//不需要添加 set 方法
//添加注入属性注解
@Autowired
//根据类型进行注入
@Qualifier
(value =
"userDaoImpl1"
)
//根据名称进行注入
private
UserDao
userDao
;
(
3
)
@Resource
:可以根据类型注入,可以根据名称注入
//@Resource //根据类型进行注入
@Resource
(name =
"userDaoImpl1"
)
//根据名称进行注入
private
UserDao
userDao
;
(
4
)
@Value
:注入普通类型属性
@Value
(value =
"abc"
)
private
String
name
;
6、完全注解开发
(
1)创建配置类,替代 xml 配置文件
@Configuration
//作为配置类,替代 xml 配置文件
@ComponentScan
(basePackages = {
"com.atguigu"
})
public class
SpringConfig {
}
(
2)编写测试类
@Test
public void
testService2() {
//加载配置类
ApplicationContext context
=
new
AnnotationConfigApplicationContext(SpringConfig.
class
);
UserService userService = context.getBean(
"userService"
,
UserService.
class
);
System.
out
.println(userService);
userService.add();
}
AOP
(概念)
1
、什么是
AOP
(
1
)面向切面编程(方面),利用
AOP
可以对业务逻辑的各个部分进行隔离,从而使得
业务逻辑各部分之间的耦合度降低,提高程序的可重用性,同时提高了开发的效率。
(2)通俗描述:不通过修改源代码方式,在主干功能里面添加新功能
(3)使用登录例子说明
AOP
1
、
AOP
底层使用动态代理
(
1
)有两种情况动态代理
第一种 有接口情况,使用
JDK
动态代理
⚫
创建接口实现类代理对象,增强类的方法
第二种 没有接口情况,使用
CGLIB
动态代理
⚫
创建子类的代理对象,增强类的方法
AOP(JDK 动态代理)
1、使用 JDK 动态代理,使用 Proxy 类里面的方法创建代理对象
(1)调用 newProxyInstance 方法
方法有三个参数:
第一参数,类加载器
第二参数,增强方法所在的类,这个类实现的接口,支持多个接口
第三参数,实现这个接口 InvocationHandler,创建代理对象,写增强的部分
2、编写 JDK 动态代理代码
(1)创建接口,定义方法
public interface UserDao {
public int add(int a,int b);
public String update(String id);
}(2)创建接口实现类,实现方法
public class UserDaoImpl implements UserDao {
@Override
public int add(int a, int b) {
return a+b;
}
@Override
public String update(String id) {
return id;
} }(3)使用 Proxy 类创建接口代理对象
public class JDKProxy {
public static void main(String[] args) {
//创建接口实现类代理对象
Class[] interfaces = {UserDao.class};
// Proxy.newProxyInstance(JDKProxy.class.getClassLoader(), interfaces,
new InvocationHandler() {
// @Override
// public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
// return null;
// }
// });
UserDaoImpl userDao = new UserDaoImpl();
UserDao dao =
(UserDao)Proxy.newProxyInstance(JDKProxy.class.getClassLoader(), interfaces,
new UserDaoProxy(userDao));
int result = dao.add(1, 2);
System.out.println("result:"+result);
} }
//创建代理对象代码
class UserDaoProxy implements InvocationHandler {
//1 把创建的是谁的代理对象,把谁传递过来
//有参数构造传递
private Object obj;
public UserDaoProxy(Object obj) {
this.obj = obj;
}
//增强的逻辑
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws
Throwable {
//方法之前
System.out.println("方法之前执行...."+method.getName()+" :传递的参
数..."+ Arrays.toString(args));
//被增强的方法执行
Object res = method.invoke(obj, args);
//方法之后
System.out.println("方法之后执行...."+obj);
return res;
} }
AOP专业术语
1、连接点
类里边那些方法可以呗增强称为连接点
2、切入点
实际被增强的方法,称为切入点
3、通知
(1)实际增强的部分称为通知(增强)
(2)通知有很多类型
前置
后置
环绕
异常
最终
4、切面
是动作 把通知应用到切入点的过程
1
、
Spring
框架一般都是基于
AspectJ
实现
AOP
操作
(
1
)
AspectJ
不是
Spring
组成部分,独立
AOP
框架,一般把
AspectJ
和
Spirng
框架一起使
用,进行
AOP
操作
2
、基于
AspectJ
实现
AOP
操作
(
1
)基于
xml
配置文件实现
(
2
)基于注解方式实现(使用)
3
、在项目工程里面引入
AOP
相关依赖
4
、切入点表达式
(
1
)切入点表达式作用:知道对哪个类里面的哪个方法进行增强
(
2
)语法结构:
execution([
权限修饰符
] [
返回类型
] [
类全路径
] [
方法名称
]([
参数列表
]) )
举例
1
:对
com.atguigu.dao.BookDao
类里面的
add
进行增强
execution(* com.atguigu.dao.BookDao.add(..))
举例
2
:对
com.atguigu.dao.BookDao
类里面的所有的方法进行增强
execution(* com.atguigu.dao.BookDao.* (..))
举例
3
:对
com.atguigu.dao
包里面所有类,类里面所有方法进行增强
execution(* com.atguigu.dao.*.* (..))
1、创建类,在类里面定义方法
public class User {
public void add() {
System.out.println("add.......");
} }2、创建增强类(编写增强逻辑)
(1)在增强类里面,创建方法,让不同方法代表不同通知类型
//增强的类
public class UserProxy {
public void before() {//前置通知
System.out.println("before......");
} }3、进行通知的配置
(1)在 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:context="http://www.springframework.org/schema/context"
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/context
http://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd">
<!-- 开启注解扫描 -->
<context:component-scan basepackage="com.atguigu.spring5.aopanno"></context:component-scan>
(2)使用注解创建
User
和
UserProxy
对象
(3)在增强类上面添加注解
@Aspect
//增强的类
@Component
@Aspect
//生成代理对象
public class
UserProxy {
(4)在
spring
配置文件中开启生成代理对象
<!-- 开启 Aspect 生成代理对象-->
<
aop
:aspectj-autoproxy
></
aop
:aspectj-autoproxy
>
4
、配置不同类型的通知
(
1
)在增强类的里面,在作为通知方法上面添加通知类型注解,使用切入点表达式配置
//增强的类
@Component
@Aspect
//生成代理对象
public class
UserProxy {
//前置通知
//@Before 注解表示作为前置通知
@Before
(value =
"execution(* com.atguigu.spring5.aopanno.User.add(..))"
)
public void
before() {
System.
out
.println(
"before........."
);
}
//后置通知(返回通知)
@AfterReturning
(value =
"execution(*
com.atguigu.spring5.aopanno.User.add(..))"
)
public void
afterReturning() {
System.
out
.println(
"afterReturning........."
);
}
//最终通知
@After
(value =
"execution(* com.atguigu.spring5.aopanno.User.add(..))"
)
public void
after() {
System.
out
.println(
"after........."
);
}
//异常通知
@AfterThrowing
(value =
"execution(*
com.atguigu.spring5.aopanno.User.add(..))"
)
public void
afterThrowing() {
System.
out
.println(
"afterThrowing........."
);
}
//环绕通知
@Around
(value =
"execution(* com.atguigu.spring5.aopanno.User.add(..))"
)
public void
around(ProceedingJoinPoint proceedingJoinPoint)
throws
Throwable {
System.
out
.println(
"环绕之前........."
);
//被增强的方法执行
proceedingJoinPoint.proceed();
System.
out
.println(
"环绕之后........."
);
}
}
5
、相同的切入点抽取
//相同切入点抽取
@Pointcut
(value =
"execution(* com.atguigu.spring5.aopanno.User.add(..))"
)
public void
pointdemo() {
}
//前置通知
//@Before 注解表示作为前置通知
@Before
(value =
"pointdemo()"
)
public void
before() {
System.
out
.println(
"before........."
);
}
6
、有多个增强类多同一个方法进行增强,设置增强类优先级
(
1
)在增强类上面添加注解
@Order(
数字类型值
)
,数字类型值越小优先级越高
@Component
@Aspect
@Order
(
1
)
public class
PersonProxy
7、完全使用注解开发
(
1)创建配置类,不需要创建 xml 配置文件
@Configuration
@ComponentScan
(basePackages = {
"com.atguigu"
})
@EnableAspectJAutoProxy
(proxyTargetClass =
true
)
public class
ConfigAop {
}
AOP
操作(
AspectJ
配置文件)
1
、创建两个类,增强类和被增强类,创建方法
2
、在
spring
配置文件中创建两个类对象
<!--创建对象-->
<
bean
id=
"book"
class=
"com.atguigu.spring5.aopxml.Book"
></
bean
>
<
bean
id=
"bookProxy"
class=
"com.atguigu.spring5.aopxml.BookProxy"
></
bean
>
3
、在
spring
配置文件中配置切入点
<!--配置 aop 增强-->
<
aop
:config
>
<!--切入点-->
<
aop
:pointcut
id=
"p"
expression=
"execution(*
com.atguigu.spring5.aopxml.Book.buy(..))"
/>
<!--配置切面-->
<
aop
:aspect
ref=
"bookProxy"
>
<!--增强作用在具体的方法上-->
<
aop
:before
method=
"before"
pointcut-ref=
"p"
/>
</
aop
:aspect
>
</
aop
:config
>
JdbcTemplate(概念和准备) 1、什么是 JdbcTemplate
(1)Spring 框架对 JDBC 进行封装,使用 JdbcTemplate 方便实现对数据库操作
2、准备工作
(1)引入相关 jar 包 (2)在 spring 配置文件配置数据库连接池
<!-- 数据库连接池 --> <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource"
destroy-method="close">
<property name="url" value="jdbc:mysql:///user_db" />
<property name="username" value="root" />
<property name="password" value="root" />
<property name="driverClassName" value="com.mysql.jdbc.Driver" />
</bean> (3)配置 JdbcTemplate 对象,注入 DataSource
<!-- JdbcTemplate 对象 --> <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
<!--注入 dataSource-->
<property name="dataSource" ref="dataSource"></property>
</bean> (4)创建 service 类,创建 dao 类,在 dao 注入 jdbcTemplate 对象
* 配置文件
<!-- 组件扫描 --> <context:component-scan base-package="com.atguigu"></context:component-scan> ⚫ Service
@Service
public class BookService {
//注入 dao
@Autowired
private BookDao bookDao; }
⚫ Dao
@Repository
public class BookDaoImpl implements BookDao {
//注入 JdbcTemplate
@Autowired
private JdbcTemplate jdbcTemplate; }
JdbcTemplate 操作数据库(添加)
1、对应数据库创建实体类
2、编写 service 和 dao
(1)在 dao 进行数据库添加操作
(2)调用 JdbcTemplate 对象里面 update 方法实现添加操作
⚫ 有两个参数
⚫ 第一个参数:sql 语句
⚫ 第二个参数:可变参数,设置 sql 语句值
@Repository
public class BookDaoImpl implements BookDao {
//注入 JdbcTemplate
@Autowired
private JdbcTemplate jdbcTemplate;
//添加的方法
@Override
public void add(Book book) {
//1 创建 sql 语句
String sql = "insert into t_book values(?,?,?)";
//2 调用方法实现
Object[] args = {book.getUserId(), book.getUsername(),
book.getUstatus()};
int update = jdbcTemplate.update(sql,args);
System.out.println(update);
} }3、测试类
@Test
public void testJdbcTemplate() {
ApplicationContext context =
new ClassPathXmlApplicationContext("bean1.xml");
BookService bookService = context.getBean("bookService",
BookService.class);
Book book = new Book();
book.setUserId("1");
book.setUsername("java");
book.setUstatus("a");
bookService.addBook(book);
}
JdbcTemplate 操作数据库(修改和删除)
1、修改
@Override
public void updateBook(Book book) {
String sql = "update t_book set username=?,ustatus=? where user_id=?";
Object[] args = {book.getUsername(), book.getUstatus(),book.getUserId()};
int update = jdbcTemplate.update(sql, args);
System.out.println(update);
}2、删除
@Override
public void delete(String id) {
String sql = "delete from t_book where user_id=?";
int update = jdbcTemplate.update(sql, id);
System.out.println(update);
}
JdbcTemplate 操作数据库(查询返回某个值)
1、查询表里面有多少条记录,返回是某个值
2、使用 JdbcTemplate 实现查询返回某个值代码
⚫ 有两个参数 ⚫ 第一个参数:sql 语句
⚫ 第二个参数:返回类型 Class
//查询表记录数
@Override
public int selectCount() {
String sql = "select count(*) from t_book";
Integer count = jdbcTemplate.queryForObject(sql, Integer.class);
return count;
}
JdbcTemplate 操作数据库(查询返回对象)
1、场景:查询图书详情
2、JdbcTemplate 实现查询返回对象
⚫ 有三个参数 ⚫ 第一个参数:sql 语句
⚫ 第二个参数:RowMapper 是接口,针对返回不同类型数据,使用这个接口里面实现类完成
数据封装
⚫ 第三个参数:sql 语句值
//查询返回对象
@Override
public Book findBookInfo(String id) {
String sql = "select * from t_book where user_id=?";
//调用方法
Book book = jdbcTemplate.queryForObject(sql, new
BeanPropertyRowMapper<Book>(Book.class), id);
return book;
}
JdbcTemplate 操作数据库(查询返回集合)
1、场景:查询图书列表分页… 2、调用 JdbcTemplate 方法实现查询返回集合
⚫ 有三个参数 ⚫ 第一个参数:sql 语句
⚫ 第二个参数:RowMapper 是接口,针对返回不同类型数据,使用这个接口里面实现类完成
数据封装
⚫ 第三个参数:sql 语句值
//查询返回集合
@Override
public List<Book> findAllBook() {
String sql = "select * from t_book";
//调用方法
List<Book> bookList = jdbcTemplate.query(sql, new
BeanPropertyRowMapper<Book>(Book.class));
return bookList;
}
JdbcTemplate 操作数据库(批量操作)
1、批量操作:操作表里面多条记录
2、JdbcTemplate 实现批量添加操作
⚫ 有两个参数 ⚫ 第一个参数:sql 语句
⚫ 第二个参数:List 集合,添加多条记录数据
//批量添加
@Override
public void batchAddBook(List<Object[]> batchArgs) {
String sql = "insert into t_book values(?,?,?)";
int[] ints = jdbcTemplate.batchUpdate(sql, batchArgs);
System.out.println(Arrays.toString(ints));
}
//批量添加测试
List<Object[]> batchArgs = new ArrayList<>();
Object[] o1 = {"3","java","a"};
Object[] o2 = {"4","c++","b"};
Object[] o3 = {"5","MySQL","c"};
batchArgs.add(o1);
batchArgs.add(o2);
batchArgs.add(o3);
//调用批量添加
bookService.batchAdd(batchArgs);
3、JdbcTemplate 实现批量修改操作
//批量修改
@Override
public void batchUpdateBook(List<Object[]> batchArgs) {
String sql = "update t_book set username=?,ustatus=? where user_id=?";
int[] ints = jdbcTemplate.batchUpdate(sql, batchArgs);
System.out.println(Arrays.toString(ints));
}
//批量修改
List<Object[]> batchArgs = new ArrayList<>();
Object[] o1 = {"java0909","a3","3"};
Object[] o2 = {"c++1010","b4","4"};
Object[] o3 = {"MySQL1111","c5","5"};
batchArgs.add(o1);
batchArgs.add(o2);
batchArgs.add(o3);
//调用方法实现批量修改
bookService.batchUpdate(batchArgs);
4、JdbcTemplate 实现批量删除操作
//批量删除
@Override
public void batchDeleteBook(List<Object[]> batchArgs) {
String sql = "delete from t_book where user_id=?";
int[] ints = jdbcTemplate.batchUpdate(sql, batchArgs);
System.out.println(Arrays.toString(ints));
}
//批量删除
List<Object[]> batchArgs = new ArrayList<>();
Object[] o1 = {"3"};
Object[] o2 = {"4"};
batchArgs.add(o1);
batchArgs.add(o2);
//调用方法实现批量删除
bookService.batchDelete(batchArgs);
事务操作(事务概念)
1
、什么事务
(
1
)事务是数据库操作最基本单元,逻辑上一组操作,要么都成功,如果有一个失败所有操
作都失败
(
2
)典型场景:银行转账
* lucy
转账
100
元 给
mary
* lucy
少
100
,
mary
多
100
2
、事务四个特性(
ACID
)
(
1
)原子性
(2)一致性
(3)隔离性
(4)持久性
1
、创建数据库表,添加记录
create table book(id int primary key atuo_increment ,username varchar(20),money int)
2
、创建
service
,搭建
dao
,完成对象创建和注入关系
(
1
)
service
注入
dao
,在
dao
注入
JdbcTemplate
,在
JdbcTemplate
注入
DataSource
@Service
public class
UserService {
//注入 dao
@Autowired
private
UserDao
userDao
;
}
@Repository
public class
UserDaoImpl
implements
UserDao {
@Autowired
private
JdbcTemplate
jdbcTemplate
;
}
3
、在
dao
创建两个方法:多钱和少钱的方法,在
service
创建方法(转账的方法)
@Repository
public class
UserDaoImpl
implements
UserDao {
@Autowired
private
JdbcTemplate
jdbcTemplate
;
//lucy 转账 100 给 mary
//少钱
@Override
public void
reduceMoney() {
String sql =
"update t_account set money=money-? where username=?"
;
jdbcTemplate
.update(sql,
100
,
"lucy"
);
}
//多钱
@Override
public void
addMoney() {
String sql =
"update t_account set money=money+? where username=?"
;
jdbcTemplate
.update(sql,
100
,
"mary"
);
}
}
@Service
public class
UserService {
//注入 dao
@Autowired
private
UserDao
userDao
;
//转账的方法
public void
accountMoney() {
//lucy 少 100
userDao
.reduceMoney();
//mary 多 100
userDao
.addMoney();
}
}
4
、上面代码,如果正常执行没有问题的,但是如果代码执行过程中出现异常,有问题
事务操作(
Spring
事务管理介绍)
1
、事务添加到
JavaEE
三层结构里面
Service
层(业务逻辑层)
2
、在
Spring
进行事务管理操作
(
1
)有两种方式:
编程式事务管理
和声明式事务管理(使用)
3
、声明式事务管理
(
1
)基于注解方式(使用)
(2)基于
xml
配置文件方式
4
、在
Spring
进行声明式事务管理,底层使用
AOP
原理
5
、
Spring
事务管理
API
(
1
)提供一个接口,代表事务管理器,这个接口针对不同的框架提供不同的实现类\
事务操作(注解声明式事务管理)
1
、在
spring
配置文件配置事务管理器
<!--创建事务管理器-->
<
bean
id=
"transactionManager"
class=
"org.springframework.jdbc.datasource.DataSourceTransactionManager"
>
<!--注入数据源-->
<
property
name=
"dataSource"
ref=
"dataSource"
></
property
>
</
bean
>
2
、
在
spring
配置文件,开启事务注解
(
1
)在
spring
配置文件引入名称空间
tx
<
beans
xmlns=
"http://www.springframework.org/schema/beans"
xmlns:
xsi
=
"http://www.w3.org/2001/XMLSchema-instance"
xmlns:
context
=
"http://www.springframework.org/schema/context"
xmlns:
aop
=
"http://www.springframework.org/schema/aop"
xmlns:
tx
=
"http://www.springframework.org/schema/tx"
xsi
:schemaLocation=
"http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx.xsd"
>
(
2
)开启事务注解
<!--开启事务注解-->
<
tx
:annotation-driven
transaction
manager=
"transactionManager"
></
tx
:annotation-driven
>
3
、在
service
类上面(或者
service
类里面方法上面)添加事务注解
(
1
)
@Transactional
,这个注解添加到类上面,也可以添加方法上面
(2)如果把这个注解添加类上面,这个类里面所有的方法都添加事务
(3)如果把这个注解添加方法上面,为这个方法添加事务
4
、
ioslation
:事务隔离级别
(
1
)事务有特性成为隔离性,多事务操作之间不会产生影响。不考虑隔离性产生很多问题
(2)有三个读问题:脏读、不可重复读、虚(幻)读
(3)脏读:一个未提交事务读取到另一个未提交事务的数据
(4)不可重复读:一个未提交事务读取到另一提交事务修改数据
4
、
timeout
:超时时间
(
1
)事务需要在一定时间内进行提交,如果不提交进行回滚
(
2
)默认值是
-1
,设置时间以秒单位进行计算
5
、
readOnly
:是否只读
(
1
)读:查询操作,写:添加修改删除操作
(
2
)
readOnly
默认值
false
,表示可以查询,可以添加修改删除操作
(
3
)设置
readOnly
值是
true
,设置成
true
之后,只能查询
6
、
rollbackFor
:回滚
(
1
)设置出现哪些异常进行事务回滚
7
、
noRollbackFor
:不回滚
(
1
)设置出现哪些异常不进行事务回滚
事务操作(
XML
声明式事务管理)
1
、在
spring
配置文件中进行配置
第一步 配置事务管理器
第二步 配置通知
第三步 配置切入点和切面
<!--1 创建事务管理器-->
<
bean
id=
"transactionManager"
class=
"org.springframework.jdbc.datasource.DataSourceTransactionManager"
>
<!--注入数据源-->
<
property
name=
"dataSource"
ref=
"dataSource"
></
property
>
</
bean
>
<!--2 配置通知-->
<
tx
:advice
id=
"txadvice"
>
<!--配置事务参数-->
<
tx
:attributes
>
<!--指定哪种规则的方法上面添加事务-->
<
tx
:method
name=
"accountMoney"
propagation=
"REQUIRED"
/>
<!--<tx:method name="account*"/>-->
</
tx
:attributes
>
</
tx
:advice
>
<!--3 配置切入点和切面-->
<
aop
:config
>
<!--配置切入点-->
<
aop
:pointcut
id=
"pt"
expression=
"execution(*
com.atguigu.spring5.service.UserService.*(..))"
/>
<!--配置切面-->
<
aop
:advisor
advice-ref=
"txadvice"
pointcut-ref=
"pt"
/>
</
aop
:config
>
事务操作(完全注解声明式事务管理)
1
、创建配置类,使用配置类替代
xml
配置文件
@Configuration
//配置类
@ComponentScan
(basePackages =
"com.atguigu"
)
//组件扫描
@EnableTransactionManagement
//开启事务
public class
TxConfig {
//创建数据库连接池
@Bean
public
DruidDataSource getDruidDataSource() {
DruidDataSource dataSource =
new
DruidDataSource();
dataSource.setDriverClassName(
"com.mysql.jdbc.Driver"
);
dataSource.setUrl(
"jdbc:mysql:///user_db"
);
dataSource.setUsername(
"root"
);
dataSource.setPassword(
"root"
);
return
dataSource;
}
//创建 JdbcTemplate 对象
@Bean
public
JdbcTemplate getJdbcTemplate(DataSource dataSource) {
//到 ioc 容器中根据类型找到 dataSource
JdbcTemplate jdbcTemplate =
new
JdbcTemplate();
//注入 dataSource
jdbcTemplate.setDataSource(dataSource);
return
jdbcTemplate;
}
//创建事务管理器
@Bean
public
DataSourceTransactionManager
getDataSourceTransactionManager(DataSource dataSource) {
DataSourceTransactionManager transactionManager =
new
DataSourceTransactionManager();
transactionManager.setDataSource(dataSource);
return
transactionManager;
}
}