Spring初步了解到深入理解

文章目录

1.Spring

1.1简介
<!-- https://mvnrepository.com/artifact/org.springframework/spring-webmvc -->
<dependency>
    <groupId>org.springframework</groupId>
    <artifactId>spring-webmvc</artifactId>
    <version>5.2.0.RELEASE</version>
</dependency>

1.2优点
  • Spring是一个开源的免费的框架
  • Spring是一个轻量级的非入侵式的框架
  • 控制反转(IOC),面向切面编程(AOP)
  • 支持事务的处理,对框架整合的支持

总结:Spring是一个轻量级的控制反转(IOC)和面向切面编程(AOP)的框架

1.3组成

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

1.4拓展

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

  • Spring Boot

    • 快速开发的脚手架
    • 基于SpringBoot可以快速开发单个的微服务
    • 约定大于配置
  • Spring Cloud

    • SpringCloud是基于SpringBoot实现的

弊端:发展了太久之后,违背了原来的理念,配置十分繁琐

2.IOC理论推导

  • 1.UserDao接口
  • 2.UserDaoImpl实现类
  • 3.UserService业务接口
  • 4.UserServiceImpl业务实现类

在我们之前的业务中,用户的需求可能会影响我们原来的代码,我们需要根据用户的需求修改源代码!如果程序代码量非常大,修改一次的代价十分昂贵!

我们使用Set接口实现对接口的切换

   private UserDao  userDao ;
  //利用set动态实现值的注入
  public void setUserDao(UserDao userDao) {
    this.userDao = userDao;
  }
//然后在MyTest中
   ((UserSeriviceImpl) userService).setUserDao(new UserDaoMySql());

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传
外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传
外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传
外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

3.Hello Spring

下面牵扯到的地址:

https://docs.spring.io/spring-framework/docs/5.2.0.RELEASE/spring-framework-reference/

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

4.IOC创建对象的方式

1.使用无参构造创建对象,默认
2.假设我们要使用有参构造创建对象
1.下标赋值
//User.java中定义有残构造
  public User(String name){
   this.name = name;
  }
//beans.xml中
    <bean id="user" class="com.hou.pojo.User">
        <constructor-arg index="0" value="小狂"/>
    </bean>
2.类型
<!--    第二种方式不建议使用-->    
<bean id="user" class="com.hou.pojo.User">
        <constructor-arg type="java.lang.String" value="琴酒"/>
    </bean>
3.参数名
    <bean id="user" class="com.hou.pojo.User">
        <constructor-arg name="name" value="琴酒"/>
    </bean>

总结:在配置文件加载的时候,容器中管理的对象就已经初始化了

5.Spring配置

5.1别名
<alias name="user" alias="users"/>
5.2bean的配置
    
<!--     name可以取别名而且name可以取多个别名-->
<bean id="user1" class="com.hou.pojo.User1" name="user12">
<!--     在getBean使用之前就user1这个对象就被创建好了-->
        <property name="name" value="西部开源"/>
    </bean>
5.3import

import一般用于团队开发使用,它可以将多个配置文件,导入合并为一个

6.依赖注入

6.1构造器注入

前面已经说过了

6.2Set方式注入[重点]
  • 依赖注入:Set注入
    • 依赖:bean对象的创建依赖于容器
    • 注入:bean对象中所有属性由容器注入

【环境搭建】

1.复杂类型

public Address getAddress() {
    return address;
  }

  public void setAddress(Address address) {
    this.address = address;
  }
  private Address address;

2.真实测试对象

public class Student {

  private String name;
  private Address address;
  private String[] books;
  private List<String>  hobbys;
  private Map<String,String> card;
  private Set<String> games;
  private String wife;
  private Properties info;
  }

3.beans.xml

<?xml version="1.0" encoding="UTF8"?>
<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">
    <!--    这里修改最后一个https成http还是报错  不知道为什么-->
    <!--bean = 对象-->
    <!--id = 变量名-->
    <!--class = new的对象-->
    <!--property 相当于给对象中的属性设值-->

    <bean id="student" class="com.hou.pojo.Student">
<!--        第一种,普通值注入-->
        <!--        如果填写除了Name都会报错,因为没有设置name-->
        <property name="name" value="琴酒"/>

    </bean>



</beans>

4.测试类

import com.hou.pojo.Student;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class MyTest {
  public static void main(String[] args) {
    ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
    Student student = (Student) context.getBean("student");
    System.out.println(student.getName());
  }
}
6.3拓展方式注入
list这些

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

<bean id="address" class="com.hou.pojo.Address">
        <property name="address" value="成都"></property>
    </bean>
    <bean id="student" class="com.hou.pojo.Student">
<!--        第一种,普通值注入-->
        <!--        如果填写除了Name都会报错,因为没有设置name-->
        <property name="name" value="琴酒"/>
<!--        第二种注入bean注入-->
        <property name="address" ref="address"/>
<!--        数组中注入,ref-->
        <property name="books">
            <array>
                <value>红楼梦</value>
                <value>西游记</value>
                <value>三国演义</value>
            </array>
        </property>
<!--   list-->
        <property name="hobbys">
                  <list>
                      <value>听歌</value>
                      <value>敲代码</value>
                      <value>看电影</value>
                  </list>
        </property>
<!--        map-->
        <property name="card">
            <map>
                <entry key="身份证" value="1212121233333"/>
                <entry key="银行卡" value="9536323233535"/>
                <entry key="" value=""/>
            </map>
        </property>

<!--        Set-->
        <property name="games">
            <set>
                <value>LOL</value>
                <value>COC</value>
                <value>BOB</value>
            </set>
        </property>
<!--        null-->
         <property name="wife">
             <null/>
         </property>
<!--         Properties
            key = value
            key = value
            -->
        <property name="info">
            <props>
                <prop key="学号">20190525</prop>
                <prop key="性别"></prop>
                <prop key="姓名">小明</prop>
            </props>
        </property>
    </bean>

p命名
<!--    p命名空间注入可以直接注入属性-->

    <bean id="user" class="com.hou.pojo.User" p:age="18" p:name="琴酒"/>
其实C命名空间就对应构造器注入,p命名空间就对应Set方式注入
userbeans.xml(使用)
//userbeans.xml
<?xml version="1.0" encoding="UTF8"?>
<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:c="http://www.springframework.org/schema/c"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd">

<!--    现在先使用p命名-->
<!--    p命名空间注入可以直接注入属性-->
<!--    还需要弄一个无参构造-->
    <bean id="user" class="com.hou.pojo.User" p:age="18" p:name="琴酒"/>
    <bean id="user2" class="com.hou.pojo.User" c:age="20" c:name="伏特加"/>

</beans>
User.java(包装)
package com.hou.pojo;

public class User {
  private  String name;

  public User(String name, int age) {
    this.name = name;
    this.age = age;
  }

  private  int age;

  public User() {

  }

  public String getName() {
    return name;
  }

  public void setName(String name) {
    this.name = name;
  }

  public int getAge() {
    return age;
  }

  public void setAge(int age) {
    this.age = age;
  }

  @Override
  public String toString() {
    return "User{" +
            "name='" + name + '\'' +
            ", age=" + age +
            '}';
  }
}

MyTest.java(测试)
  @Test
  public  void  test2(){
//    我服了 我这里又没有配置userbeans.xml
    ApplicationContext context = new ClassPathXmlApplicationContext("userbeans.xml");
//    我们如果知道是什么对象的话可以有第二个参数
    User user = context.getBean("user2",User.class);
    System.out.println(user);
  }

注意点:p命名和c命名空间不能直接使用,需要导入xml约束

<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:c="http://www.springframework.org/schema/c"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
        http://www.springframework.org/schema/beans/spring-beans.xsd">
6.4 bean的作用域

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

1.单例模式(Spring默认机制)
<bean id="user2" class="com.hou.pojo.User" c:age="20" c:name="伏特加" scope="singleton"/>
2.原型模式

每次从容器中get,都会取得一个新的对象

<bean id="user2" class="com.hou.pojo.User" c:age="20" c:name="伏特加" scope="prototype"/>
3.其余的request,session,application,这个只能在web开发中使用到

7.Bean的自动装配

  • 自动装配是Spring满足bean依赖的一种方式!
  • Spring会在上下文中自动寻找,并给Bean装配属性

在Spring中有三种装配的方式

  • 1.xml中显示的配置
  • 2.在java中显示的配置
  • 3.隐式的自动装配
7.1测试

1.环境搭建

  • 一个人有两个宠物
7.2ByName自动装配
<!--  byName:会自动在容器上下文中查找,和自己对象set方法后面的值对应的beanid!-->  
   <bean id="person" class="com.hou.pojo.Person" autowire="byName">
        <property name="name" value="口区"></property>
    </bean>
7.3ByType自动装配
<!--
byName:会自动在容器上下文中查找,和自己对象set方法后面的值对应的beanid!
byType:会自动在容器上下文中查找,和自己对象属性类型相同的bean,是根据class来判定的
-->
    <bean  class="com.hou.pojo.Dog"/>
    <bean  class="com.hou.pojo.Cat"/>
    <bean id="person" class="com.hou.pojo.Person" autowire="byType">
        <property name="name" value="口区"></property>
    </bean>

小姐:

  • byname,需要保证所有bean的id唯一,并且这个bean需要和自动注入的set方法值一致
  • bytype的时候,需要保证所有bean的class唯一,并且这个bean需要和自动注入的属性的类型一致
7.4 使用注解实现自动装配

jdk1.5支持的注解,Spring2.5就支持注解了

使用注解须知

1.导入约束

2.配置注解的支持

重要:context:annotation-config/
<?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"
    xsi:schemaLocation="http://www.springframework.org/schema/beans
        https://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context
        https://www.springframework.org/schema/context/spring-context.xsd">

    <context:annotation-config/>

</beans>
//Person.java
    @Autowired
    private Cat cat;
    @Autowired
    private Dog dog;
    private String name;
<?xml version="1.0" encoding="UTF8"?>
<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">
    <!--    这里修改最后一个https成http还是报错  不知道为什么-->
    <!--bean = 对象-->
    <!--id = 变量名-->
    <!--class = new的对象-->
    <!--property 相当于给对象中的属性设值-->

    <!--
    byName:会自动在容器上下文中查找,和自己对象set方法后面的值对应的beanid!
    byType:会自动在容器上下文中查找,和自己对象属性类型相同的bean,是根据class来判定的
    -->
<!--    开启注解支持-->
    <context:annotation-config/>

    <bean id="dog" class="com.hou.pojo.Dog"/>
    <bean id="cat" class="com.hou.pojo.Cat"/>
    <bean id="person" class="com.hou.pojo.Person"/>

</beans>
@AutoWired

可以直接在属性上使用!也可以在Set方式上使用

使用Autowired之后就可以不用编写Set方法,前提是你这个自动装配的属性在IOC(Spring)容器中存在,且符合名字ByName

@Nullable 字段标记了这个注解,说明字段可以为空
public @interface Autowired{
  boolean required() default true;
}
  //如果显示定义了Autowired的required属性为false,说明这个对象可以为Null,佛则不允许为空
    @Autowired(required = false)
@Qualifier

如果出现多个不同的bean可以用Qualifier来指定(使用Qualifier来配合Autowired)

    <bean id="dog22" class="com.hou.pojo.Dog"/>
    <bean id="dog222" class="com.hou.pojo.Dog"/>
    <bean id="cat222" class="com.hou.pojo.Cat"/>
    <bean id="cat2222" class="com.hou.pojo.Cat"/>
//Qualifier来指定值
@Autowired
@Qualifier(value = "cat222")
private Cat cat;
@Autowired
@Qualifier(value = "dog222")
private Dog dog;
private String name
@Resource
@Resource比较智能,
@Resource( name = "cat2")时候不会报错
@Resource的时候:
1.cat的名字不同不会报错
2.cat的类名不同不会报错
小结:

Resource和Autowired的区别

  • 都是用来自动装配的,都可以放在属性字段上
  • @AutoWired 通过byType的方式实现
  • @Resource默认通过byName的方式实现,如果找不到名字,则通过byType实现
  • 执行顺序不同:@Autowired 通过byType实现

8.使用注解开发

1.bean

在Spring4之后,要使用注解开发,必须要保证aop包导入了
外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

使用注解后需要导入context约束,增加注解的支持

<?xml version="1.0" encoding="UTF8"?>
<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">
    <!--    这里修改最后一个https成http还是报错  不知道为什么-->
    <!--bean = 对象-->
    <!--id = 变量名-->
    <!--class = new的对象-->
    <!--property 相当于给对象中的属性设值-->

    <!--    开启注解支持-->
    <context:annotation-config/>
</beans>
2.属性如何注入
package com.hou.pojo;

import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;
//等价于<bean id="user" class="com.kuang.pojo.User">
@Component
public class User {
  //相当于<property name="name" value="狂神">
  @Value("狂神")
  public String name;

  public void setName(String name) {
    this.name = name;
  }
}
3.衍生的注解

@Component 有几个衍生注解,我们在web开发中,会按照Mvc三层架构分层

  • dao【@Repository】
  • service【@Service】
  • controller【@Controller】

这四个注解功能都是一样的,都是代表将某个类注册Spring容器中,装配Bean!

4.自动装配

#注解说明

@Autowired:自动装配通过类型名字
@Nullable  字段标记了这个注解,说明这个字段可以为null
@Resource:先通过名字再通过类型

5.作用域

//等价于<bean id="user" class="com.kuang.pojo.User">
@Component


@Scope("prototype")
public class User {
  //相当于<property name="name" value="狂神">
  @Value("狂神")
  public String name;

  public void setName(String name) {
    this.name = name;
  }
}

6.小结

xml与注解:

  • xml更加万能,适用于任何场合!维护简单方便
  • 注解,不是自己类使用不了,维护相对复杂!

xml与注解最佳实践:

  • xml用来管理bean
  • 注解只负责完成属性的
  • 我们在使用的过程中,只需要注意一个问题,必须让注解生效,就需要开启注解的支持

7.各个注解的作用

1@controller 控制器(注入服务) ---相当于<bean id="UserController" class="com.hou.controller">
2@service 服务(注入dao)  ---想当于<bean id="UserService" class="com.hou.service">
3@repository dao(实现dao访问)
4@component (把普通pojo实例化到spring容器中,相当于配置文件中的<bean id="user" class="com.hou.pojo.User"/>5.@Value("kuangshen2")
  //在xml中的扫描组件
   <context:component-scan base-package="com.hou"/>
    <!--    开启注解支持-->
    <context:annotation-config/>

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

9.使用Java的方式配置Spring

我们现在要完全不适用Spring的xml配置,全权交给Java来做

JavaConfig是Spring的一个子项目,在Spring 4之后,它成为了一个核心功能

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

//实体类
package com.hou.pojo;

import org.springframework.beans.factory.annotation.Value;

public class User {
  public String getName() {
    return name;
  }

  @Value("琴酒")    //属性注入值

  public void setName(String name) {
    this.name = name;
  }

  private String name;

  @Override
  public String toString() {
    return "User{" +
            "name='" + name + '\'' +
            '}';
  }
}

-------------------------------------------------------

  //配置文件
  package com.hou.config;

import com.hou.pojo.User;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.Import;


//这里注解的意思就是这个类被Spring接管了,注册到了容器中

  //在一个类上只要加入了Configuration就类似于在外面套了一个beans
  @Configuration    //这个Configuration也会被Spring容器托管,注册到容器中,因为他本来就是一个@Component
  //怎么说呢,Configuration就代表一个配置类,就和我们之前遇到的beans.xml是一样的
 @ComponentScan("com.hou.pojo")
  @Import(HouConfig2.class)
  //这个和原来XML中的    <context:component-scan base-package="com.hou.pojo"/>是一样的
  public class HouConfig {
    //这一句就相当于在Bean里面写了一个bean文件了
    //这个方法的名字,就相当于bean标签中的id属性
    //这个方法的返回值,就相当于Bean标签中的class属性
    @Bean
    public User user(){
      return new User();
     //就是返回要注入bean的对象
    }
  }


}
-----------------------------------------------------
  //测试类
import com.hou.config.HouConfig;
import com.hou.pojo.User;
import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;

public class MyTest {
  public static void main(String[] args) {
//


    //注解里面加载的是一个类
    //如果按照配置类去做,就只能通过AnnotationConfig上下文来获取容器
    ApplicationContext context = new AnnotationConfigApplicationContext(HouConfig.class);

    User getUser = (User) context.getBean("user");
    System.out.println(getUser.getName());


//    System.out.println("asdasdsa");
  }
}

10、代理模式

代理模式是SpringAOP的底层

代理模式:

  • 静态代理
  • 动态代理

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

10.1 静态代理

1.抽象角色:一般会使用接口或抽象类解决

package com.hou.demo;

public interface Rent {
  public void  Rent();
}

2.真实角色:被代理的角色

package com.hou.demo;

public class Host implements Rent{
  public void Rent() {
    System.out.println("房东正在出租房子");
  }
}

3.代理角色:代理真实角色

package com.hou.demo;

public class Proxy implements Rent{
  private Host host;

  public Proxy() {
  }

  public Proxy(Host host) {
    this.host = host;
  }
  //这里因为代理是帮房东租房子所以应该用host.rent();
  public void Rent() {
    host.Rent();
    seeHouse();
    fare();
  }

  //看房
  public  void seeHouse(){
    System.out.println("中介带你看房");
  }
  //中介收费
  public  void fare(){
    System.out.println("收中介费");
  }
}

4.客户:访问代理对象的人

package com.hou.demo;

public class Client {
  public static void main(String[] args) {
    Host host = new Host();
    //代理
    //需要向这里传参
    //在构造哪里设置了有参数构造
    Proxy proxy = new Proxy(host);
    proxy.Rent();

  }
}

代理模式的好处:

  • 可以使真实角色的操作更加纯粹
  • 公共也就交给了代理角色,实现了业务的分工
  • 公共业务发生拓展的时候,方便集中管理

缺点:一个真实角色就会产生一个代理角色,代码量会翻倍,开发效率会降低(动态代理应该就是为了解决这个问题)

10.2加深理解
10.3动态代理
  • 动态代理和静态代理角色一样
  • 动态代理的代理类是动态生成的,不是我们直接写的
  • 动态代理分为两类:
    • 基于接口的动态代理:JDK动态代理
    • 基于类的动态代理:cglib
    • java字节码

Proxy(代理)

InvocationHanler:调用处理程序

Interface InvocationHandler
  • InvocationHandler是由代理实例的调用处理程序实现的接口

    每个代理实例都有一个关联的调用处理程序。 当在代理实例上调用方法时,方法调用将被编码并分派到其调用处理程序的invoke方法。

Class Proxy
反射

JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意一个方法和属性;这种动态获取的信息以及动态调用对象的方法的功能称为java语言的反射机制。

  • 可以使真实角色的操作更加纯粹
  • 公共也就交给了代理角色,实现了业务的分工
  • 公共业务发生拓展的时候,方便集中管理
  • 一个动态代理类代理一个接口,一般就是一类业务
  • 一个动态代理类可以代理多个类,只要实现一个接口既可以
ServiceImpl
//ServiceImpl
package com.hou.demo2;

public class UserServiceImpl implements UserService{

  public void add() {
    System.out.println("增加了一个用户");
  }

  public void delete() {
    System.out.println("删除了一个用户");
  }

  public void update() {
    System.out.println("修改了一个用户");
  }

  public void select() {
    System.out.println("查询了一个用户");
  }
}

Test
package com.hou.demo4;

import com.hou.demo2.UserService;
import com.hou.demo2.UserServiceImpl;

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.add();

  }
}

ProxyInvocationHandler
package com.hou.demo4;


import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

//等会我们会用这个类自动生成代理类
public class ProxyInvocationHandler implements InvocationHandler {

//  Foo f = (Foo) Proxy.newProxyInstance(Foo.class.getClassLoader(),
//          new Class<?>[] { Foo.class },
//          handler);

  //被代理的接口

  private  Object target;

  public void setTarget(Object target) {
    this.target = target;
  }

  //得到生成代理对象
  public Object getProxy(){
    //一个类加载器,一个接口,一个InvacationHanler
     return Proxy.newProxyInstance(this.getClass().getClassLoader(),target.getClass().getInterfaces(),this);
  }

  //处理代理实例并返回结果
  public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
    log(method.getName());
    //动态代理的本质,就是发射机制的实现
    Object result =method.invoke(target,args);

    return  result;

  }
  public  void log(String msg){
    System.out.println("执行"+msg+"方法");
  }

}

11.AOP

11.1 什么是AOP

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传
外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

使用AOP需要导入的依赖
dependency>
            <groupId>org.aspectj</groupId>
            <artifactId>aspectjrt</artifactId>
            <version>1.9.6</version>
        </dependency>
        <dependency>
            <groupId>org.aspectj</groupId>
            <artifactId>aspectjweaver</artifactId>
            <version>1.9.6</version>
        </dependency>
        <dependency>
            <groupId>aopalliance</groupId>
            <artifactId>aopalliance</artifactId>
            <version>1.0</version>
</dependency>
11.3使用Spring实现扫描AOP
方式一:使用API接口【主要是SpringAPI接口的实现】
UserServiceImpl提供信息
package com.hou.service;

public class UserServiceImpl implements UserService{

  public void add() {
    System.out.println("增加了一个用户");
  }

  public void delete() {
    System.out.println("删除了一个用户");
  }

  public void update() {
    System.out.println("更新了一个用户");
  }

  public void select() {
    System.out.println("选择了一个用户");
  }
}

Log日志当执行前出入
package com.hou.log;

import org.springframework.aop.MethodBeforeAdvice;

import java.lang.reflect.Method;

public class Log implements MethodBeforeAdvice {

  //methods:要执行的目标对象的方法
  //args:参数
  //target:目标对象
  public void before(Method method, Object[] args, Object target) throws Throwable {
    System.out.println(target.getClass().getName()+"的"+method.getName()+"被执行了");
  }
}

AfterLog日志 在执行后
package com.hou.log;

import org.springframework.aop.AfterReturningAdvice;

import java.lang.reflect.Method;

public class AfterLog  implements AfterReturningAdvice {

  //returnValue 返回值
  public void afterReturning(Object returnValue, Method method, Object[] args, Object target) throws Throwable {
    System.out.println("执行了"+method.getName()+"方法,返回结果为"+returnValue);
  }
}

AOP负责连接
<?xml version="1.0" encoding="UTF8"?>
<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 = 对象-->
    <!--id = 变量名-->
    <!--class = new的对象-->
    <!--property 相当于给对象中的属性设值-->
<!--    出现找不到这个xml需要点右上角的设置让这个xml重新加载-->

    <!--    注册bean-->
    <bean id="userService" class="com.hou.service.UserServiceImpl"/>
    <bean id="log" class="com.hou.log.Log"/>
    <bean id="afterlog" class="com.hou.log.AfterLog"/>
    <!--    使用Spring原生API接口-->
    <!--    配置aop:需要导入aop的约束-->
    <aop:config>
        <!--        切入点: expression:表达式  execution(要执行的位置)  修饰词 返回值 类名 方法名 参数-->

        <aop:pointcut id="pointcut" expression="execution(* com.hou.service.UserServiceImpl.*(..))"/>
        <!--        执行环绕增加-->
        <!--        把log这个类切入到这个方法上面-->
        <!--        因为log和afterlog里面使用的是利用反射的动态管理,所以可以用动态管理来管理接口-->
        <aop:advisor advice-ref="log" pointcut-ref="pointcut"/>
        <aop:advisor advice-ref="afterlog" pointcut-ref="pointcut"/>

    </aop:config>

</beans>
最后的测试类
import com.hou.service.UserService;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class MyTest {
  public static void main(String[] args) {
    ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
    //动态代理代理的是接口
    UserService userService = (UserService) context.getBean("userService");
    userService.add();
  }
}
方式二:使用自定义类【主要是切面定义】
自定义了一个Java文件
package com.hou.diy;

public class DiyPointCut {
  public void before(){
    System.out.println("=======方法执行前=========");
  }
  public void after(){
    System.out.println("=======方法执行后=========");
  }
}

在xml中做了一些修改

    <aop:config>
        <aop:aspect ref="diy">
<!--            切入点-->
<!--            我认为pointcut应该是被载入,pointcut-ref应该是载入-->
         <aop:pointcut id="point" expression="execution(* com.hou.service.UserServiceImpl.*(..))"/>
<!--            通知--><aop:before method="before" pointcut-ref="point"/>
                       <aop:after method="after" pointcut-ref="point"/>
        </aop:aspect>
    </aop:config>
方式三:使用注解开发AOP
新建一个类使用注解
package com.hou.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;
@Aspect  //标注这个类是一个切面
public class AnnotationPointCut {
  @Before("execution(* com.hou.service.UserServiceImpl.*(..))")
  public void before(){
    System.out.println("=======方法执行前=========");
  }
  @After("execution(* com.hou.service.UserServiceImpl.*(..))")
  public void after(){
    System.out.println("=======方法执行后=========");
  }
  @Around("execution(* com.hou.service.UserServiceImpl.*(..))")
  public void around(ProceedingJoinPoint jp) throws Throwable {
    System.out.println("环绕前");
    //执行方法
    Object proceed =jp.proceed();
    System.out.println("环绕后");
//    Signature signature = jp.getSignature();  //获得签名
//    System.out.println("signature:"+signature);

//    System.out.println(proceed);
  }
}

xml导入bean
<!--    方式三-->
    <bean id="annotationPointCut" class="com.hou.diy.AnnotationPointCut"/>
<!--    开启注解支持 JDK(默认 proxy-target-class="false") cglib(proxy-target-class="true")-->
     <aop:aspectj-autoproxy/>

12.整合MyBatis

步骤:

1.导入jar包

  • junit
  • mybatis
  • mysql数据库
  • spring相关
  • aop织入
  • mybatis-spring【new】

2.编写配置文件

3.测试

12.1回忆Mybatis

1.编写实体类

2.编写核心配置文件

3.编写接口

4.编写Mapper

5.测试