IOC
1、什么是IOC
(1)控制反转,把对象创建和对象之间的调用过程交给spring进行管理
(2)使用IOC目的:为了降低耦合度
2、IOC底层原理
(1)XML解析、工厂模式、反射
3、画图讲解工厂模式
class UserService {
execute ( ) {
UserDao dao = new UserDao ( ) ;
dao. add ( ) ;
}
}
class UserDao ( ) {
add ( ) {
. . . . . .
}
}
= > 上面的程序耦合度太高了,一旦UserDao 改变,则UserService 也要跟着改变
工厂模式:
class UserService {
execute ( ) {
UserDao dao = UserFactory . getDao ( ) ;
dao. add ( ) ;
}
}
class UserDao {
add ( ) {
. . . . . .
}
}
工厂类:
class UserFactory {
public static UserDao getDao ( ) {
return new UserDao ( ) ;
}
}
= > 通过工厂模式降低了耦合度
= 》目的:耦合度降低最低限度
IOC过程
第一步:XML配置文件,配置创建的对象
<bean id="dao" class="com.xxx.UserDao"></bean>
第二步:有service类和dao类,创建工厂类
class UserFactory {
public static UserDao getDao ( ) {
String classValue = class 属性值;
Class clazz = Class . forName ( classValue)
return ( UserDao ) clazz. newInstance ( ) ;
}
}
IOC(接口)
1、IOC思想基于IOC容器完成,IOC容器底层就是对象工厂
2、Spring提供IOC容器实现两种方式:(两个接口)
(1)BeanFactory:IOC容器基本实现,是Spring内部的使用接口,不提供开发者进行使用
*加载配置文件时候不会创建对象,在获取对象才去创建
(2)ApplicationContext:BeanFactory接口的子接口,提供了更多更强大的功能,一般有开发者进行使用
*加载配置文件时候就会把在配置文件对象进行创建
IOC操作Bean管理(概念)
1、什么是Bean管理
(1)Bean管理指的是两个操作
(2)Spring创建对象
(3)Spring注入属性
2、Bean管理操作有两种方式
(1)基于XML配置文件方式实现
(2)基于注解方式实现
IOC操作Bean管理(基于XMl方式)
1、基于XML方式创建对象
<bean id="user" class="com.xxx.User"></bean>
(1)在Spring配置文件中,使用bean标签,标签里面添加对应属性。就可以实现对象创建
(2)在bean标签有很多属性,介绍常用属性
*id属性:唯一标识
*class属性:类全路径(包类路径)
(3)创建对象时候,默认也是执行无参构造方法完成对象创建
2、基于XML方式注入属性
(1)DI:依赖注入,就是注入属性
3、第一种注入方式:使用set方法进行注入
(1)创建类:定义类的属性和对应的set方法
public class Book(){
private String bName;
public void setName(String bName){
this.bName = bName
}
}
(2)在spring配置文件配置对象创建,配置属性注入
<bean id="book" class="com.xxx.Book">
<property name="bName" value="鸡你太美"></property>
</bean>
4、第二种注入方式:使用有参构造方法进行注入
(1)创建类,定义属性,创建属性对应有参数构造方法
public class Orders {
private String name;
private String address;
//有参数构造
public Orders(String name,String address) {
this.name = name;
this.address = address;
}
}
(2)在 spring 配置文件中进行配置
有参构造注入属性
<bean id="order" class="com.example.Order">
<constructor-arg name="name" value="鸡你太美"></constructor-arg>
<constructor-arg name="address" value="坤坤家"></constructor-arg>
</bean>
IOC操作Bean管理(XML注入其他类型属性)
1、字面量
(1)null值
<propert name="address">
<null/>
</propert>
(2)属性值包含特殊符号:可以用转义字符,也可以用CDATA进行注入
<property name="BName">
<value><![CDATA[<<鸡你太美>>]]]></value>
</property>
2、注入属性-外部bean
(1)创建两个类service类和mapper类
(2)在service调用dao里面的方法
(3)在Spring配置文件中进行配置
service类:
public class UserService {
private UserMapper userMapper;
public void setUserMapper ( UserMapper userMapper) {
this . userMapper = userMapper;
}
public void add ( ) {
System . out. println ( "service add..." ) ;
userMapper. update ( ) ;
}
}
mapper类:
public class UserMapper {
public void update ( ) {
System . out. println ( "UserUpdate..." ) ;
}
}
bean:
< bean id = " userService" class = " com.example.service.UserService" >
< property name = " userMapper" ref = " userMapper" > </ property>
</ bean>
< bean id = " userMapper" class = " com.example.mapper.UserMapper" > </ bean>
3、注入属性-内部bean和级联赋值
(1)一对多关系:部门和员工
一个不萌有多个员工,一个员工属于一个部门
部门是一,员工是多
(2)在实体类之间表示一对多关系,员工表示所属部门,使用对象类型属性进行表示
部门类
public class Dept {
private String name;
public void setName(String name) {
this.name = name;
}
@Override
public String toString() {
return "Dept{" +
"name='" + name + '\'' +
'}';
}
}
员工类:
public class Emp {
private String name;
private String gender;
private Dept dept;
public void setDept(Dept dept) {
this.dept = dept;
}
public void setName(String name) {
this.name = name;
}
public void setGender(String gender) {
this.gender = gender;
}
@Override
public String toString() {
return "Emp{" +
"name='" + name + '\'' +
", gender='" + gender + '\'' +
", dept=" + dept +
'}';
}
}
(3)在spring配置文件中进行配置
<bean id="emp" class="com.example.domain.Emp">
<property name="name" value="坤坤"></property>
<property name="gender" value="鸡"></property>
<property name="dept">
<bean id="dept" class="com.example.domain.Dept">
<property name="name" value="坤部"></property>
</bean>
</property>
</bean>
(4)注入属性-级联赋值
第一种写法:
<bean id="emp" class="com.example.domain.Emp">
<property name="name" value="坤坤"></property>
<property name="gender" value="鸡"></property>
<property name="dept" ref="dept"></property>
</bean>
<bean id="dept" class="com.example.domain.Dept">
<property name="name" value="坤部"></propert>
</bean>
第二种写法:
<bean id="emp" class="com.example.domain.Emp">
<property name="name" value="坤坤"></property>
<property name="gender" value="鸡"></property>
<property name="dept" ref="dept"></property>
<property name="dept.name" ref="坤部"></property>
</bean>
<bean id="dept" class="com.example.domain.Dept"></bean>
IOC操作Bean管理(XML注入集合属性)
1、注入数组类型属性
2、注入List集合类型属性
3、注入Map集合类型属性
4、注入Set集合类型属性
(1)创建类,定义数组、list、map、set 类型属性,生成对应 set 方法
public class Stu {
private String[] courses;
private List<String> list;
private Map<String, String> map;
private Set<String> set;
public void setCourses(String[] courses) {
this.courses = courses;
}
public void setList(List<String> list) {
this.list = list;
}
public void setMap(Map<String, String> map) {
this.map = map;
}
public void setSet(Set<String> set) {
this.set = set;
}
}
(2)在Spring配置文件中进行配置
<bean id="stu" class="com.example.domain.Stu">
<property name="courses">
<array>
<value>Java</value>
<value>MySql</value>
</array>
</property>
<property name="list">
<list>
<value>坤坤</value>
<value>鸡哥</value>
</list>
</property>
<property name="map">
<map>
<entry key="JAVA" value="java"/>
<entry key="MYSQL" value="mysql"/>
</map>
</property>
<property name="set">
<set>
<value>Redis</value>
<value>Spring</value>
</set>
</property>
</bean>
5、在集合里面设置对象类型值
<bean name="course1" class="com.example.domain.Course">
<property name="name" value="Spring5框架"/>
</bean>
<bean name="course2" class="com.example.domain.Course">
<property name="name" value="计算机组成原理"/>
</bean>
<property name="courseList">
<list>
<ref bean="course1"/>
<ref bean="course2"/>
</list>
</property>
IOC操作Bean管理(FactoryBean)
1、Spring有两种类型bean,一种普通bean,另外一种工厂bean(FactoryBean)
2、普通bean:在配置文件中定义bean类型就是返回类型
3、工厂bean:在配置文件中定义bean类型可以和返回类型不一样
第一步:创建类,让这个类作为工厂bean,实现接口FactoryBean
第二步:实现接口方法,爱实现的方法中定义返回的bean类型
定义FactoryBean
public class MyBean implements FactoryBean<Course> {
//定义返回bean
@Override
public Course getObject() throws Exception {
Course course = new Course();
course.setName("abc");
return course;
}
@Override
public Class<?> getObjectType() {
return null;
}
@Override
public boolean isSingleton() {
return FactoryBean.super.isSingleton();
}
}
在Spring配置文件中进行配置:
<bean id="myBean" class="com.example.factoryBean.MyBean">
测试案例:
@Test
public void Test6(){
ApplicationContext context = new ClassPathXmlApplicationContext("bean.xml");
Course myBean = context.getBean("myBean", Course.class);
System.out.println(myBean);
}
IOC操作Bean管理(bean作用域)
1、在Spring里面,设置创建bean实例时单例还是多例
2、在Spring里面,默认情况下是单例。
@Test
public void Test1() {
ApplicationContext context = new ClassPathXmlApplicationContext("bean.xml");
Book book1 = context.getBean("book", Book.class);
Book book2 = context.getBean("book", Book.class);
System.out.println(book1);
System.out.println(book2);
System.out.println(book1.hashCode());
System.out.println(book2.hashCode());
}
结果:
com.example.domain.Book@358c99f5
com.example.domain.Book@358c99f5
898406901
898406901
3、如何设置单实例还是多实例
(1)在Spring配置文件bean标签里面有属性(scope)用于设置单例还是多例
(2)scope属性值
第一个值:默认值,singleton,表示单例对象
第二个值:prototype,表示多例对象
<bean id="book" class="com.example.domain.Book" scope="prototype"></bean>
结果:
com.example.domain.Book@358c99f5
com.example.domain.Book@3ee0fea4
898406901
1054932644
(3)singleton和prototype区别
第一:singleton单例,prototype多例
第二:scope是singleton时,加载spring配置文件时就会创建单例对象
scope是prototype时,在调用getBean方法时就会创建多例对象
IOC操作Bean管理(bean生命周期)
1、生命周期:从对象的创建到销毁的过程
2、bean生命周期(创建=》赋值=》初始化=》调用=》销毁)
=》 通过无参构造方法创建bean实例
=》 为bean设置属性值和对其他bean引用(调用set方法)
=》 调用bean的初始化方法(需要配置初始化方法)
=》 bean可以使用(对象获取到了)
=》 当容器关闭时候,调用bean的销毁方法(需要配置销毁方法)
3、演示bean的生命周期
public class Order2 {
private String name;
public Order2() {
System.out.println("第一步:执行了无参构造方法创建bean实例");
}
public void setName(String name) {
this.name = name;
System.out.println("第二步:调用set方法设置属性值");
}
//创建执行的初始化的方法
public void initMethod() {
System.out.println("第三步:执行初始化的方法");
}
//创建执行的销毁的方法
public void destroyMethod() {
System.out.println("第五步:执行销毁的方法");
}
}
<bean id="order2" class="com.example.domain.Order2" init-method="initMethod" destroy-method="destroyMethod">
<property name="name" value="手机"/>
</bean>
@Test
public void Test7(){
ApplicationContext context = new ClassPathXmlApplicationContext("bean.xml");
Order2 order2 = context.getBean("order2", Order2.class);
System.out.println("第四步:获取创建的bean实例对象");
System.out.println(order2);
//手动让bean实例销毁
((ClassPathXmlApplicationContext) context).close();
}
结果:
第一步:执行了无参构造方法创建bean实例
第二步:调用set方法设置属性值
第三步:执行初始化的方法
第四步:获取创建的bean实例对象
com.example.domain.Order2@4b168fa9
第五步:执行销毁的方法
4、bean的后置处理器,bean的生命周期有七步(创建=》赋值 =》pro =》初始化 =》 pro =》调用=》销毁)
=》 通过无参构造方法创建bean实例
=》 为bean设置属性值和对其他bean引用(调用set方法)
=》 把bean实例传递bean后置处理器的方法 -> postProcessBeforeInitialization
=》 调用bean的初始化方法(需要配置初始化方法)
=》 把bean实例传递bean后置处理器的方法 -> postProcessAfterInitialization
=》 bean可以使用(对象获取到了)
=》 当容器关闭时候,调用bean的销毁方法(需要配置销毁方法)
5、演示添加后置处理器效果
(1)创建类,实现接口BeanPostProcessor,创建后置处理器
public class MyBeanPost implements BeanPostProcessor {
@Override
public Object postProcessBeforeInitialization(Object bean, String beanName) throws BeansException {
System.out.println("在初始化前执行的方法");
return bean;
}
@Override
public Object postProcessAfterInitialization(Object bean, String beanName) throws BeansException {
System.out.println("在初始化后执行的方法");
return bean;
}
}
<!-- 配置后置处理器 -> 会对所有的bean加上后置处理器的方法-->
<bean id="myBeanPost" class="com.example.domain.MyBeanPost"></bean>
结果:
第一步:执行了无参构造方法创建bean实例
第二步:调用set方法设置属性值
在初始化前执行的方法
第三步:执行初始化的方法
在初始化后执行的方法
第四步:获取创建的bean实例对象
com.example.domain.Order2@7920ba90
第五步:执行销毁的方法
IOC操作Bean管理(XML自动装配)
1、什么是自动装配:根据指定装配规则(属性名或者属性类型),Spring自动将匹配的属性值进行注入
2、演示自动装配过程
(1)根据属性名称自动注入装配
<bean id="emp2" class="com.example.domain.Emp2" autowire="byName"></bean>
<bean id="dept2" class="com.example.domain.Dept2"></bean>
(2)根据属性类型自动注入装配
<bean id="emp2" class="com.example.domain.Emp2" autowire="byType"></bean>
<bean id="dept2" class="com.example.domain.Dept2"></bean>
IOC操作Bean管理 (外部属性文件)
1、直接配置数据库信息
(1)配置Druid连接池
<bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource">
<property name="driverClassName" value="com.mysql.jdbc.Driver"/>
<property name="url" value="jdbc:mysql://localhost:3306/userDb"/>
<property name="username" value="root"/>
<property name="password" value="root"/>
</bean>
(2)引入Druid连接池配置jar包
2、引入外部属性文件配置数据库连接池
prop.driverClassName=com.mysql.jdbc.Driver
prop.url=jdbc:mysql://localhost:3306/userDb
prop.driverClassName=root
prop.driverClassName=root
IOC操作Bean管理(基于注解方式)
1、什么是注解:
(1)注解是代码特殊标记,格式:@注解名称(属性名称=属性值,属性名称=属性值...)
(2)使用注解,注解作用在类上面,方法上面,属性上面
(3)使用注解目的:简化XML配置
2、Spring针对Bean管理中创建对象提供注解
(1)@Component
(2)@Service
(3)@Controller
(4)@Repository
*上面四个注解功能是一样的,都可以用来创建bean实例
3、基于注解方式实现对象创建
第一步:引入依赖
第二步:开启注解扫描
<context:component-scan base-package="com.example"></context:component-scan>
第三步:创建类,在类上面添加注解‘
@Service
public class UserService {
public void add(){
System.out.println("UserService add...");
}
}
4、基于注解方式实现属性注入
(1)@AutiWired:根据属性类型进行自动注入
第一步 把 service 和 mapper 对象创建,在 service 和 mapper 类添加创建对象注解
第二步 在 service 注入 mapper 对象,在 service 类添加 apper 类型属性,在属性上面使用注解
@Service
public class UserService {
@Autowired
private UserMapper userMapper;
public void add() {
System.out.println("UserService add...");
userMapper.add();
}
}
(2)@Qualifier:根据属性名称进行注入
这个@Qualifier注解的使用,和上面@Autowired一起使用
(3)@Resource:可以根据类型注入,也可以根据名称注入
(4)@Value:注入普通类型属性
AOP
1、什么是AOP
(1)面向切面编程,利用AOP可以对业务逻辑的各个部分进行隔离,从而是业务逻辑各部分之间的耦合度降低,
提高程序的可重用性,同时提高了开发的效率。
(2)通俗描述:不通过修改源代码方式,在主干功能里面添加新功能。
(3)使用登录例子说明AOP
AOP底层原理
1、AOP底层使用动态代理
(1)有两种情况动态代理
第一种:有接口情况,使用JDK动态代理
*创建接口实现类代理对象,增强类方法
第二种:没有接口情况,使用CGLIB动态代理
AOP(JDK动态代理)
1、使用JDK动态代理,使用Proxy类里面的方法创建代理对象
java.lang.reflect.Proxy
(1)调用newProxyInstance方法
方法有三个参数:
第一个参数:类加载器
第二个参数:增强方法所在的类,这个类实现的接口。支持多接口
第三个参数:实现这个接口InvocationHandler,创建代理对象,写增强的部分
2、编写JDK动态代理代码
(1)创建接口
public interface UserMapper {
int add(int a, int b);
String update(String id);
}
(2)创建接口实现类,实现方法
public class UserMapperImpl implements UserMapper {
@Override
public int add(int a, int b) {
return a + b;
}
@Override
public String update(String id) {
return id;
}
}
(3)使用Proxy类创建代理对象
AOP(术语)
1、连接点:类里面有哪些方法可以被增强,这些方法称为连接点
2、切入点:实际被增强的方法,称为切入点
3、通知(增强):
(1)实际增强的逻辑部分称为通知(增强)
(2)通知的类型
*前置通知
*后置通知
*环绕通知
*异常通知
*最终通知
4、切面:是通知,把通知应用到切入点的过程
AOP操作(准备)
1、Spring框架一般都是基于AspectJ实现AOP操作
(1)什么是AspectJ:ASpectJ不是Spring组成部分,独立AOP框架,一把AspectJ和Spring框架一起使用,进行AOP操作
2、基于AspectJ实现AOP操作
(1)基于XML配置文件实现
(2) 基于注解实现
3、引入依赖
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-aop</artifactId>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-aspects</artifactId>
</dependency>
4、切入点表达式
(1)切入点表达式作用:知道对那个类里面的哪个方法进行增强
(2)语法结构:execution([权限修饰符][返回类型][类全路径][方法名称]([参数列表]))
eg:
Ⅰ、对com.xxx.mapper.BookMapper类里面的add进行增强
execution(* com.xxx.mapper.BookMapper.add(..))
Ⅱ、对com.xxx.mapper.BookMapper类里面的所有方法进行增强
execution(* com.xxx.mapper.BookMapper.*(..))
Ⅲ、对com.xxx.mapper包里面所有类,类里面所有方法进行增强
execution(* com.xxx.mapper.*.*(..))
AOP操作(AspectJ注解)
1、创建类,在类里面定义方法
public class User {
public void add(){
System.out.println("add......");
}
}
2、创建增强类(编写增强逻辑)
(1)在增强类里面,创建方法,让不同方法代表不同通知类型
public class UserProxy {
public void add(){
System.out.println("before.....");
}
}
3、进行通知的配置
(1)在Spring配置文件中,开启注解扫描
(2)使用注解创建User和UserProxy对象
(3)在增强类上面添加注解@Aspect
(4)在spring配置文件中开启生成代理对象
4、配置不同类型的通知
(1)在增强类的里面,在作为通知方法上面添加通知类型注解,使用切入点表达式配置
@Component
@Aspect
public class UserProxy {
//前置通知
@Before(value = "execution(* com.example.annoAop.User.add(..))")
public void before(){
System.out.println("before.....");
}
//最终通知
@After(value = "execution(* com.example.annoAop.User.add(..))")
public void affter(){
System.out.println("after......");
}
//后置通知
@AfterReturning(value = "execution(* com.example.annoAop.User.add(..))")
public void afterReturning(){
System.out.println("afterReturning......");
}
//异常通知
@AfterThrowing(value = "execution(* com.example.annoAop.User.add(..))")
public void afterThrowing(){
System.out.println("afterThrowing......");
}
//环绕通知
@Around(value = "execution(* com.example.annoAop.User.add(..))")
public void around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable {
System.out.println("aroundBefore......");
proceedingJoinPoint.proceed();
System.out.println("aroundBefore......");
}
}
顺序:环绕前 =》 前置 =》 本方法 =》 后置 =》 环绕后 =》 异常 =》 最终
其中(后置与环绕后)跟 异常二选一
5、相同切入点抽取
(1)提取相同切入点
@Pointcut(value = "execution(* com.example.annoAop.User.add(..))")
public void point(){
}
(2)在通知加切入点
@Before(value = "point()")
public void before(){
System.out.println("before.....");
}
6、有多个增强类,设置优先级
在增强类上面添加注解 @Order(数字类型值),数字类型值越小优先级越高
@Component
@Aspect()
@Order(1)
public class PersonProxy {
@Before("execution(* com.example.annoAop.User.add(..))")
public void before(){
System.out.println("我是Person before");
}
}