Spring初步了解到深入理解

news2024/11/18 11:33:15

文章目录

      • 1.Spring
        • 1.1简介
        • 1.2优点
        • 1.3组成
        • 1.4拓展
      • 2.IOC理论推导
      • 3.Hello Spring
        • 下面牵扯到的地址:
      • 4.IOC创建对象的方式
        • 1.使用无参构造创建对象,默认
        • 2.假设我们要使用有参构造创建对象
          • 1.下标赋值
          • 2.类型
          • 3.参数名
      • 5.Spring配置
        • 5.1别名
        • 5.2bean的配置
        • 5.3import
      • 6.依赖注入
        • 6.1构造器注入
        • 6.2Set方式注入\[重点]
        • 6.3拓展方式注入
          • list这些
          • p命名
          • **其实C命名空间就对应构造器注入,p命名空间就对应Set方式注入**
          • userbeans.xml(使用)
          • User.java(包装)
          • MyTest.java(测试)
        • 6.4 bean的作用域
          • 1.单例模式(Spring默认机制)
          • 2.原型模式
          • 3.其余的request,session,application,这个只能在web开发中使用到
      • 7.Bean的自动装配
        • 7.1测试
        • 7.2ByName自动装配
        • 7.3ByType自动装配
        • 7.4 使用注解实现自动装配
          • 重要:<context:annotation-config/>
          • **@AutoWired**
          • @Qualifier
          • @Resource
          • 小结:
      • 8.使用注解开发
        • 1.bean
        • 2.属性如何注入
        • 3.衍生的注解
      • 9.使用Java的方式配置Spring
      • 10、代理模式
        • 10.1 静态代理
        • 10.2加深理解
        • 10.3动态代理
        • Interface InvocationHandler
        • Class Proxy
        • 反射
          • ServiceImpl
          • Test
          • ProxyInvocationHandler
      • 11.AOP
        • 使用AOP需要导入的依赖
        • 11.3使用Spring实现扫描AOP
        • 方式一:使用API接口【主要是SpringAPI接口的实现】
          • UserServiceImpl提供信息
          • Log日志当执行前出入
          • AfterLog日志 在执行后
          • AOP负责连接
          • 最后的测试类
        • 方式二:使用自定义类【主要是切面定义】
          • 自定义了一个Java文件
        • 方式三:使用注解开发AOP
          • 新建一个类使用注解
          • xml导入bean
      • 12.整合MyBatis
        • 12.1回忆Mybatis

1.Spring

1.1简介
  • Spring—>给软件行业带来了春天
  • Spring是一个轻量级的控制反转(IOC)和面向切面(AOP)的容器框架
  • 2002,首次推出了Spring框架的雏形
  • Spring框架以interface21框架为基础,经过重新设计,并不断丰富内涵,与2004年发布1.0正式版
  • SSH:Struct2+Spring+Hibernate
  • SSM: SpringMVC + Spring + Mybatis
  • 官网:https://spring.io/projects/spring-framework
  • 下载地址:https://repo.spring.io/release/org/springframework/spring/
  • GitHub地址:https://github.com/spring-projects/spring-boot
<!-- 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.lang.Object
  • java.lang.reflect.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.测试

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1129422.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Monocular arbitrary moving object discovery and segmentation 代码复现

环境 https://github.com/michalneoral/Raptor 1.创建environment.yaml name: raptor channels:- pytorch- conda-forge dependencies:- python3.8- pytorch1.9.0- torchvision0.10.0- cudatoolkit11.1- pipconda env create -f environment.yaml conda activate raptor2.安…

C++基类和派生类的内存分配,多态的实现

目录 基类和派生类的内存分配基类和派生类的成员归属多态的实现 基类和派生类的内存分配 类包括成员变量&#xff08;data member&#xff09;和成员函数&#xff08;member function&#xff09;。 成员变量分为静态数据&#xff08;static data&#xff09;和非静态数据&…

技术分享 | 针对蜜罐反制Goby背后的故事

0x01 概述 近期我们联动FORadar做了一个插件&#xff0c;实现了从企业名称->企业漏洞的全自动检测流程&#xff0c;在做具体实践的时候碰到了两个很有意思的蜜罐&#xff0c;其中一个蜜罐内置了Weblogic漏洞&#xff0c;同时配置有专门针对旧版本Goby反制Payload&#xff0…

点亮现代编程语言的男人——C语言/UNIX之父Dennis Ritchie

祝各位程序员们1024程序员节快乐&#x1f389;&#x1f389;&#x1f389; 图片来自网络&#xff0c;侵删 前言 在程序员中&#xff0c;有一位人物的不被人熟知&#xff0c;他的贡献甚至比他自身更要出名 C语言之父&#xff0c;UNIX之父——Dennis MacAlistair Ritchie 一…

0基础学习PyFlink——使用Table API实现SQL功能

在《0基础学习PyFlink——使用PyFlink的Sink将结果输出到Mysql》一文中&#xff0c;我们讲到如何通过定义Souce、Sink和Execute三个SQL&#xff0c;来实现数据读取、清洗、计算和入库。 如下图所示SQL是最高层级的抽象&#xff0c;在它之下是Table API。本文我们会将例子中的SQ…

【机器学习合集】深度学习模型优化方法最优化问题合集 ->(个人学习记录笔记)

文章目录 最优化1. 最优化目标1.1 凸函数&凹函数1.2 鞍点1.3 学习率 2. 常见的深度学习模型优化方法2.1 随机梯度下降法2.2 动量法(Momentum)2.3 Nesterov accelerated gradient法(NAG)2.4 Adagrad法2.5 Adadelta与Rmsprop法2.6 Adam法2.7 Adam算法的改进 3. SGD的改进算法…

LVS+keepalived高可用集群

1、定义 keepalived为lvs应运而生的高可用服务。lvs的调度器无法做高可用&#xff0c;keepalived实现的是调度器的高可用&#xff0c;但keepalived不只为lvs集群服务的&#xff0c;也可以做其他代理服务器的高可用&#xff0c;比如nginxkeepalived也可实现高可用&#xff08;重…

解密Kubernetes:探索开源容器编排工具的内核

&#x1f90d; 前端开发工程师&#xff08;主业&#xff09;、技术博主&#xff08;副业&#xff09;、已过CET6 &#x1f368; 阿珊和她的猫_CSDN个人主页 &#x1f560; 牛客高级专题作者、在牛客打造高质量专栏《前端面试必备》 &#x1f35a; 蓝桥云课签约作者、已在蓝桥云…

zabbix6.0 部署配置

架构 先简单介绍zabbix监控的最主要的两个组件&#xff1a; zabbix server zabbix agent server 用来部署 web console以及相关的数据存储&#xff0c;所以需要配合一些数据库来保存数据&#xff0c;比如mysql,pgsql, 又有前端的页面所以还需要配置 nginx 和getway 所以 serve…

【makedown使用介绍】

如何使用makedown 欢迎使用Markdown编辑器新的改变功能快捷键合理的创建标题,有助于目录的生成如何改变文本的样式插入链接与图片如何插入一段漂亮的代码片生成一个适合你的列表创建一个表格设定内容居中、居左、居右SmartyPants创建一个自定义列表如何创建一个注脚注释也是必…

计算机网络【CN】IPV4报文格式

版本&#xff08;4bit&#xff09;&#xff1a;IPV4/IPV6首部长度&#xff08;4bit&#xff09;&#xff1a;标识首部的长度 单位是4B最小为&#xff1a;20B最大为&#xff1a;60&#xff08;15*4&#xff09;B总长度&#xff08;16bit&#xff09;&#xff1a;整个数据报&…

目录遍历漏洞

漏洞挖掘之目录遍历漏洞 (baidu.com) 从0到1完全掌握目录遍历漏洞 0x01 什么是目录遍历漏洞 目录遍历漏洞是由于网站存在配置缺陷&#xff0c;导致网站目录可以被任意浏览&#xff0c;这会导致网站很多隐私文件与目录泄露。 比如数据库备份文件、配置文件等&#xff0c;攻击…

Vue项目中使用require的方式导入图片资源,本地运行无法打开的问题

问题描述 项目经理说需快速要写一个大屏&#xff0c;然后拿给售前去给客户做个展示。其中有一块需要展示一个拓扑图&#xff0c;绘制拓扑图时用了定义了一个图片节点&#xff0c;然后图片的导入方式是 require的方式&#xff0c;然后本地npm run dev启动的时候可以正常显示&…

JVM进阶(1)

一)JVM是如何运行的&#xff1f; 1)在程序运行前先将JAVA代码转化成字节码文件也就是class文件&#xff0c;JVM需要通过类加载器将字节码以一定的方式加载到JVM的内存运行时数据区&#xff0c;将类的信息打包分块填充在运行时数据区&#xff1b; 2)但是字节码文件是JVM的一套指…

大数据技术学习笔记(二)—— Hadoop 运行环境的搭建

目录 1 准备模版虚拟机hadoop1001.1 修改主机名1.2 修改hosts文件1.3 修改IP地址1.3.1 查看网络IP和网关1.3.2 修改IP地址 1.4 关闭防火墙1.5 创建普通用户1.6 创建所需目录1.7 卸载虚拟机自带的open JDK1.8 重启虚拟机 2 克隆虚拟机3 在hadoop101上安装JDK3.1 传输安装包并解压…

likeadmin部署

以下内容写于2023年9月17日&#xff0c;likeadmin版本 1.登录页404&#xff0c;且无法登录 参照官方教程部署后&#xff0c;访问登录页&#xff0c;能打开但提示404&#xff0c;点登录也是404&#xff0c;在issues中搜到新搭建的环境&#xff0c;登录管理后台&#xff0c;报re…

系统设计 - 我们如何通俗的理解那些技术的运行原理 - 第八部分:Linux、安全

本心、输入输出、结果 文章目录 系统设计 - 我们如何通俗的理解那些技术的运行原理 - 第八部分&#xff1a;Linux、安全前言Linux 文件系统解释应该知道的 18 个最常用的 Linux 命令HTTPS如何工作&#xff1f;数据是如何加密和解密的&#xff1f;为什么HTTPS在数据传输过程中会…

java通过IO流下载保存文件

我们在开发过程中&#xff0c;可能会遇到需要到远程服务器上下载文件的需求&#xff0c;一般我们的文件可能会有一个url地址&#xff0c;我们拿到这个地址&#xff0c;可以构建URLConnection对象&#xff0c;之后可以根据这个URLConnection来获取InputStream&#xff0c;之后&a…

C++ list 的使用

目录 1. 构造函数 1.1 list () 1.2 list (size_t n, const T& val T()) 1.3 list (InputIterator first, InputIterator last) 2. bool empty() const 3. size_type size() const 4. T& front() 4. T& back() 5. void push_front (const T& val) 6.…

【Java系列】Java 基础

目录 基础1.JDK和JRE的区别2.Java为什么不直接实现lterator接口&#xff0c;而是实现lterable?3.简述什么是值传递和引用传递?4.概括的解释下Java线程的几种可用状态? 中级1.简述Java同步方法和同步代码块的区别 ?2.HashMap和Hashtable有什么区别?3.简述Java堆的结构? 什…