DAY04_Spring—Aop案例引入代理机制

目录

  • 1 AOP
    • 1.1 AOP案例引入
      • 1.1.1 数据库事务说明
    • 1.2 Spring实现事务控制
      • 1.2.1 代码结构如下
      • 1.2.2 编辑User
      • 1.2.3 编辑UserMapper/UserMapperImpl
      • 1.2.4 编辑UserService/UserServiceImpl
      • 1.2.5 编辑配置类
      • 1.2.6 编辑测试类
    • 1.3 代码问题分析
    • 1.4 代理模式
      • 1.4.1 生活中代理案例
      • 1.4.2 代理模式
    • 1.5 静态代理
      • 1.5.1 通过代理模式实现事务控制
      • 1.5.2 修改目标对象名称
      • 1.5.3 编辑代理类
      • 1.5.4 编辑测试类
    • 1.5 静态代理弊端
    • 1.6 动态代理机制
      • 1.6.1 动态代理分类
      • 1.6.2 编辑JDK动态代理
      • 1.6.3 JDK动态代理执行过程
    • 1.7 动态代理优势
      • 1.7.1 编辑DeptService/DeptServiceImpl
      • 1.7.2 编辑测试类
    • 1.8 动态代理实现Demo2
      • 1.8.1 业务需求
      • 1.8.2 编辑UserService/UserServiceImpl
      • 1.8.3 编辑代理工厂
      • 1.8.4 编辑测试案例

1 AOP

1.1 AOP案例引入

1.1.1 数据库事务说明

  • 案例分析:
    • userMapper.insert(User对象)
    • deptMapper.insert(dept对象)
      • 由于业务需求 要求方法要么同时入库,要么同时回滚.所以必须通过事务进行控制.

1.2 Spring实现事务控制

1.2.1 代码结构如下

在这里插入图片描述

1.2.2 编辑User

public class User {private Integer id;private String name;public Integer getId() {return id;}public void setId(Integer id) {this.id = id;}public String getName() {return name;}public void setName(String name) {this.name = name;}@Overridepublic String toString() {return "User{" +"id=" + id +", name='" + name + '\'' +'}';}
}

1.2.3 编辑UserMapper/UserMapperImpl

  • 编辑UserMapper
import com.jt.pojo.User;public interface UserMapper {void addUser(User user);
}
  • 编辑UserMapperImpl
import com.jt.pojo.User;
import org.springframework.stereotype.Repository;@Repository
public class UserMapperImpl implements UserMapper {//??事务控制应该在哪一层完成!!dao/mapper service@Overridepublic void addUser(User user) {System.out.println("用户入库"+user);}
}

1.2.4 编辑UserService/UserServiceImpl

  • 编辑UserService
import com.jt.pojo.User;public interface UserService {void addUser(User user);
}
  • 编辑UserServiceImpl
import com.jt.mapper.UserMapper;
import com.jt.pojo.User;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;@Service
public class UserServiceImpl implements UserService{@Autowiredprivate UserMapper userMapper;//事务控制应该放在Service层中控制@Overridepublic void addUser(User user) {try {System.out.println("事务开始");userMapper.addUser(user);System.out.println("事务结束");} catch (Exception e) {e.printStackTrace();System.out.println("事务回滚");}}
}

1.2.5 编辑配置类

import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;@Configuration//标识我是一个配置类
@ComponentScan("com.jt")
public class SpringConfig {
}

1.2.6 编辑测试类

import com.jt.config.SpringConfig;
import com.jt.pojo.User;
import com.jt.service.UserService;
import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.AnnotationConfigApplicationContext;public class TestUser {//Controller-Service-Mapper(Dao)//Spring中规定://  如果传入的是接口的类型 则自动查找/注入 该接口的实现类//  该接口只有一个实现类@Testpublic void testTx(){ApplicationContext context = new AnnotationConfigApplicationContext(SpringConfig.class);//写接口的类型/实现类的类型?UserService userService = context.getBean(UserService.class);
//        UserService userService = (UserService) context.getBean("userServiceImpl");User user = new User();user.setId(101);user.setName("SpringAop测试入门案例");userService.addUser(user);}
}

1.3 代码问题分析

  • Service层应该写代码业务,但是现在与事务控制紧紧的耦合在一起
  • 代码冗余不便于大批量开发

在这里插入图片描述

  • 解决方案:
    • 采用代理模式进行编辑.

1.4 代理模式

1.4.1 生活中代理案例

  • 房屋中介代理模式:
    • 房东: 自己手里有房子 需要出租换钱
    • 中介机构 1.本职工作 带客户看房/出租房屋 2.收取中介费(服务费)
    • 租客: 满足自身需求 租房子
  • 代码思维建模:
    • 暴露一个公共的接口(租房子)
    • 客户与中介机构进行沟通,中介看起来和房东功能一致.(代理看起来就是真实的对象)
    • 完成用户额外的操作(收取中介费)

1.4.2 代理模式

  • 组成部分
    • 要求代理者实现与被代理者相同的接口
    • 在代理方法中实现功能的扩展
    • 用户调用代理对象完成功能(用户认为代理就是目标对象)
  • 调用流程

在这里插入图片描述

1.5 静态代理

1.5.1 通过代理模式实现事务控制

  • 角色划分:
    • 目标对象target UserServiceImpl类
    • 目标方法 method addUser()方法
    • 代理: 实现事务控制.
    • 代理对象与目标对象实现相同的接口.

1.5.2 修改目标对象名称

在这里插入图片描述

1.5.3 编辑代理类

import com.jt.pojo.User;
import com.jt.service.UserService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;@Service("userService")
public class StaticProxy implements UserService {//要求引入目标对象@Autowired //ByType  byName//@Qualifier("target")private UserService target;//目的: 对原有方法进行扩展@Overridepublic void addUser(User user) {try {System.out.println("事务开始");target.addUser(user);System.out.println("事务结束");} catch (Exception e) {e.printStackTrace();System.out.println("事务回滚");}}
}

1.5.4 编辑测试类

@Test
public void testStaticProxy(){ApplicationContext context = new AnnotationConfigApplicationContext(SpringConfig.class);UserService userService = (UserService) context.getBean("userService");User user = new User();user.setId(10001);user.setName("测试代理机制");//执行用户调用userService.addUser(user);
}

1.5 静态代理弊端

  • 静态代理只针对于某个接口 不能实现所有接口的代理 实用性较差

在这里插入图片描述

  • 静态代理中所有的方法,都需要手动的添加事务开始/事务提交代码 代码冗余 不够简洁

在这里插入图片描述

1.6 动态代理机制

1.6.1 动态代理分类

  • JDK代理:
    • 要求: 要求目标对象必须实现接口
    • 代理要求: 代理对象也必须实现目标对象的接口
    • 目标对象/代理关系: 目标对象与代理对象兄弟关系.
  • CGlib代理
    • 要求: 不管目标对象是否有接口,都可以为其创建代理对象
    • 代理要求: 要求代理对象必须继承目标对象
    • 目标对象/代理关系: 目标对象与代理对象是父子关系

在这里插入图片描述

1.6.2 编辑JDK动态代理

官网API:

在这里插入图片描述

  • 知识点:
    • 关于匿名内部类用法说明: 匿名内部类引用外部参数 要求参数必须final修饰
    • 该方法标识 当代理对象执行时,"回调"该方法.
       public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {}
    
    • 目标方法执行
    result = method.invoke(target,args);
    
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;//能否利用一个工厂动态为目标对象创建代理
public class JDKProxyFactory {//要求用户传递目标对象//关于匿名内部类用法说明:  匿名内部类引用外部参数 要求参数必须final修饰public static Object getProxy(final Object target) {//1.调用java API实现动态代理/***  参数分析: 3个参数*      1.ClassLoader loader, 类加载器(获取目标对象的Class)*      2.类<?>[] interfaces,  JDK代理要求 必须有接口*                             java中可以多实现*      3.InvocationHandler h  对目标方法进行扩展*///1.获取类加载器ClassLoader classLoader = target.getClass().getClassLoader();//2.获取接口数组Class[] interfaces = target.getClass().getInterfaces();//3.通过动态代理创建对象Object proxy = Proxy.newProxyInstance(classLoader, interfaces, new InvocationHandler() {//invoke方法: 代理对象调用方法时invoke执行,扩展方法的编辑位置@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {//proxy: 代理对象本身//method: 用户调用的方法对象//args:   用户调用方法的参数// result 标识目标方法执行的返回值Object result = null;try {//添加事务的控制System.out.println("事务开始");//执行目标方法// target真实的目标对象,method方法对象,args方法参数result = method.invoke(target, args);System.out.println("事务提交");} catch (Exception e) {e.printStackTrace();System.out.println("事务回滚");}return result;}});return proxy;}
}
  • 测试代码
/*** 测试JDK动态代理*/@Testpublic void testJDKProxy(){ApplicationContext context = new AnnotationConfigApplicationContext(SpringConfig.class);//1.获取用户目标对象UserService target = (UserService) context.getBean("target");//2.获取代理对象UserService userService = (UserService) JDKProxyFactory.getProxy(target);//3.打印代理对象的类型System.out.println(userService.getClass());//4.用户完成调用User user = new User();user.setId(10001);user.setName("JDK动态代理完成");//执行用户调用userService.addUser(user);}

在这里插入图片描述

1.6.3 JDK动态代理执行过程

在这里插入图片描述

1.7 动态代理优势

将公共的部分写到动态代理中,之后其他的业务类调用即可

1.7.1 编辑DeptService/DeptServiceImpl

  • 编辑DeptService
public interface DeptService {void addDept();
}
  • 编辑DeptServiceImpl
import org.springframework.stereotype.Service;@Service("deptService")
public class DeptServiceImpl implements DeptService {@Overridepublic void addDept() {//添加事务System.out.println("调用DeptMapper实现入库操作");//提交事务}
}

1.7.2 编辑测试类

@Testpublic void testTx() {//1.获取目标对象ApplicationContext context = new AnnotationConfigApplicationContext(SpringConfig.class);DeptService target = (DeptService) context.getBean("deptService");//2.获取代理对象DeptService deptService = (DeptService) JDKProxyFactory.getProxy(target);//通过代理对象 调用方法  扩展了方法!!!!!deptService.addDept();  //invoke}

在这里插入图片描述

1.8 动态代理实现Demo2

1.8.1 业务需求

  • 要求:
    • 对Service层的方法记录其执行的时间
    • service中 有 addUser方法/deleteUser方法.
    • 要求代码结构扩展性好,耦合性低.

1.8.2 编辑UserService/UserServiceImpl

  • 编辑UserService
public interface UserService {void addUser();void deleteUser();
}
  • 编辑UserServiceImpl
@Service("target")
public class UserServiceImpl implements UserService{@Overridepublic void addUser() {System.out.println("新增用户");}@Overridepublic void deleteUser() {System.out.println("删除用户");}
}

1.8.3 编辑代理工厂

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;public class JDKProxyFactory {//编辑静态方法获取代理对象public static Object getProxy(final Object target) {//3个参数  1.类加载器  2.对象的接口ClassLoader classLoader = target.getClass().getClassLoader();Class[] interfaces = target.getClass().getInterfaces();Object proxy = Proxy.newProxyInstance(classLoader, interfaces, new InvocationHandler() {//代理对象执行目标方法时执行@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {//让用户执行目标方法Long startTime = System.currentTimeMillis(); //开始时间//执行目标方法 获取返回值 可能为nullObject result = method.invoke(target);Long endTime = System.currentTimeMillis();  //结束时间System.out.println("程序执行:" + (endTime - startTime) + "毫秒");//将返回值传递给调用者return result;}});return proxy;}
}

1.8.4 编辑测试案例

@Testpublic void test01(){ApplicationContext context = new AnnotationConfigApplicationContext(SpringConfig.class);//1.获取目标对象UserService target = (UserService) context.getBean("target");//2.获取代理对象UserService proxy = (UserService) JDKProxyFactory.getProxy(target);System.out.println(proxy.getClass());//3.调用业务方法proxy.addUser();proxy.deleteUser();}

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

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

JCIM | pointsite(点云)在蛋白结合位点预测方面的应用

这篇论文介绍了一种名为PointSite的创新方法&#xff0c;用于准确识别蛋白质结构中的配体结合位点&#xff08;LBS&#xff09;&#xff0c;相较于先前的技术具有更高的准确性。PointSite采用了蛋白质为中心的方法&#xff0c;而不是仅仅依赖于蛋白质结构外的伪表面点&#xff…

protobuf学习日记 | 认识protobuf中的类型

目录 前言 一、标量数据类型 二、protobuf中的 “数组” 三、特殊类型 1、枚举类型 &#xff08;1&#xff09;类型讲解 &#xff08;2&#xff09;升级通讯录 2、Any类型 &#xff08;1&#xff09;类型讲解 &#xff08;2&#xff09;升级通讯录 3、oneof类型 …

实验笔记之——基于TUM-RGBD数据集的SplaTAM测试

之前博客对SplaTAM进行了配置&#xff0c;并对其源码进行解读。 学习笔记之——3D Gaussian SLAM&#xff0c;SplaTAM配置&#xff08;Linux&#xff09;与源码解读-CSDN博客SplaTAM全称是《SplaTAM: Splat, Track & Map 3D Gaussians for Dense RGB-D SLAM》&#xff0c;…

九、K8S-label和label Selector

label和label selector 标签和标签选择器 1、label 标签&#xff1a; 一个label就是一个key/value对 label 特性&#xff1a; label可以被附加到各种资源对象上一个资源对象可以定义任意数量的label同一个label可以被添加到任意数量的资源上 2、label selector 标签选择器 L…

C++中特殊类的设计与单例模式的简易实现

设计一个只能在堆上创建对象的类 对于这种特殊类的设计我们一般都是优先考虑私有构造函数。然后对于一些特殊要求就直接通过静态成员函数的实现来完成。 class A//构造函数私有&#xff08;也可以析构函数私有&#xff09; { public:static A* creat(){return new A;} privat…

【Redis】基于Token单点登录

基于Token单点登录 获取验证码 流程图 #mermaid-svg-DLGHgCofEYXVSmI5 {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-DLGHgCofEYXVSmI5 .error-icon{fill:#552222;}#mermaid-svg-DLGHgCofEYXVSmI5 .error-text{f…

Oracle行转列函数,列转行函数

Oracle行转列函数&#xff0c;列转行函数 Oracle 可以通过PIVOT,UNPIVOT,分解一行里面的值为多个列,及来合并多个列为一行。 PIVOT PIVOT是用于将行数据转换为列数据的查询操作(类似数据透视表)。通过使用PIVOT&#xff0c;您可以按照特定的列值将数据进行汇总&#xff0c;并将…

【webrtc】GCC 7: call模块创建的ReceiveSideCongestionController

webrtc 代码学习&#xff08;三十二&#xff09; video RTT 作用笔记 从call模块说起 call模块创建的时候&#xff0c;会创建 src\call\call.h 线程&#xff1a; 统计 const std::unique_ptr<CallStats> call_stats_;SendDelayStats &#xff1a; 发送延迟统计 const…

用Go plan9汇编实现斐波那契数列计算

斐波那契数列是一个满足递推关系的数列&#xff0c;如&#xff1a;1 1 2 3 5 8 ... 其前两项为1&#xff0c;第3项开始&#xff0c;每一项都是其前两项之和。 用Go实现一个简单的斐波那契计算逻辑 func fib(n int) int {if n 1 || n 2 {return 1}return fib(n-1) fib(n-2) …

RT-Thread experimental 代码学习(1)thread_sample

RTOS的最基础功能是线程。 线程的调度是如何工作的&#xff1f;RT-thread官方的实验文档是最好的参考。 老规矩&#xff0c;先放法国人doxygen。 thread_sample 代码的调用关系图 有意思的是&#xff0c;RT有两种创建线程的方式 - 静态和动态&#xff0c;粗略的理解是&…

Qt弹框展示

1.相关说明 文件选择弹框、目录选择弹框、保存文件弹框、颜色选择弹框、字体选择弹框、进度条弹框、输入对话框、标准消息框等 2.相关界面 3.相关代码 #include "widget.h" #include "ui_widget.h" #include <QFileDialog> #include <QProgressD…

Windows下安装alipay-sdk-python时,pycrypto安装报错问题处理

1、安装alipay-sdk-python 时&#xff0c;保存内容如下。 Building wheels for collected packages: pycryptoBuilding wheel for pycrypto (setup.py) ... error error: subprocess-exited-with-error python setup.py bdist_wheel did not run successfully.│ exit c…