mybatis之动态代理的应用

在前文(https://www.cnblogs.com/NYfor2018/p/9093472.html)我们知道了,Mybatis的使用需要用到Mapper映射文件,一个是映射接口,另一个是映射XML文件(此处不详谈映射文件XML),在应用中我们可以感觉到,映射接口似乎对接着XML文件中的实现命令,可是我们在运行程序是时候调用的往往是Mapper接口,而不是一个包含逻辑的实现类。很显然Mapper产生了代理类。

首先,什么是代理模式?代理模式的定义:为其他对象提供一种代理以控制对这个对象的访问。在某些情况下,一个对象不适合或者不能直接引用另一个对象,而代理对象可以在客户端和目标对象之间起到中介的作用。(取自百度百科:https://baike.baidu.com/item/%E4%BB%A3%E7%90%86%E6%A8%A1%E5%BC%8F/8374046?fr=aladdin

举个栗子,如图:

我们租房的时候一般是去找中介,而不是直接去找包租婆。放在代理模式这里来说,就是,我们在访问真实的对象的时候,往往不是直接去访问真实对象,而是通过代理对象,来对真实对象进行访问。

为什么要使用代理模式?通过代理,一方面可以控制如何访问真正的服务对象,提供额外的服务。另外一方面有机会通过重写一些类来满足特定的需要。就像是,有时候租客反映的一些问题,中介可以直接解决而不需要麻烦到包租婆。

一般来说,动态代理分为两种:一种是JDK反射机制提供的代理;另一种是CGLB代理。在JDK提供的代理,我们必须要提供接口;而在CGLIB中则不需要提供接口。

在学习动态代理之前,先了解一下反射的基础。

反射简单应用

import java.lang.reflect.Method;
public class ReflectService { public void sayHello(String name) { System.out.println("hello"+name); } public static void main(String[] args) throws Exception, IllegalAccessException, ClassNotFoundException { //通过反射创建ReflectService对象 Object service = Class.forName(ReflectService.class.getName()).newInstance(); //获取服务方法 Method method = service.getClass().getMethod("sayHello", String.class); //反射调用方法 method.invoke(service, "zhangsan");
}
}

 

①   先根据类名,来创建实例对象,所以就找了ReflectService类的类名。

②   然后再根据实例对象来找回它的方法,参数是方法名及其参数。

③   利用反射机制来调用ReflectService类的sayHello方法。

JDK动态代理

 

JDK的动态代理,是由JDK的java.lang.reflect.*包提供支持的,我们需要完成以下步骤:

① 编写服务类和接口,服务类是真正的服务提供者,而JDK代理中接口是必要的。

② 编写代理类,提供绑定和代理方法。

首先,先编写动态代理的接口

public interface HelloService {

         public void sayHello(String name);

}

接着,写一个这个接口的实现类

public class HelloServiceImpl implements HelloService{

         @Override
public void sayHello(String name) {
System.out.println("hello "+name);
}
}

然后,写一个代理类,提供真实对象的绑定和代理方法。代理类的要求是实现InvocationHandler接口的代理方法,当一个对象被绑定后,执行其方法的时候就会进入到代理方法里。

import java.lang.reflect.InvocationHandler;

import java.lang.reflect.Method;

import java.lang.reflect.Proxy;

public class HelloServiceProxy implements InvocationHandler{

         /**

          * 真实的服务对象

          */

         private Object target;

         /**

          *

          * @param target

          * @return 绑定委托对象并返回一个代理类

          */

         public Object bind(Object target) {

                   this.target = target;

                   //取得代理对象

                   //public static Object newProxyInstance(ClassLoader loader,Class<?>[] interfaces,InvocationHandler h)throws IllegalArgumentException

                   //loader服务对象的类加载器,interfaces是加载服务对象的接口,h是执行这个代理类的方法的执行者

                   return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), this);

         }

         @Override

         /**

          * 通过代理对象调用方法首先进入这个方法

          * invoke方法的参数分别是:proxy是代理对象,method是被调用的方法,args是方法的参数

          */

         public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {

                   System.err.println("#########我是JDK动态代理##########");

                   Object result = null;

                   //反射方法前调用

                   System.err.println("我准备说hello");

                   result = method.invoke(target, args);

                   //反射方法后调用

                   System.err.println("我说过hello了");

                   return result;

         }

}

最后,编写一个测试类:

public class HelloServiceMain {

         public static void main(String[] args) {

                   HelloServiceProxy HelloHandler = new HelloServiceProxy();

                   HelloService proxy = (HelloService)HelloHandler.bind(new HelloServiceImpl());

                   proxy.sayHello("张三");

         }

}

运行出来的结果是:

整个过程可以这样理解:

先创建一个代理类对象a,然后用代理类对象绑定真正提供服务对象b,返回一个接口b+,再用这个接口b+来进行服务。

如果我们把proxy.sayHello(“张三”);注释掉:

我们会发现控制台什么东西都没有输出。

所以我们可以知道,代理类的invoke方法,只有在代理的真正提供服务的对象被调用的时候,才会起作用。

所以这时候我们可以这样理解:

挑选的人相当于访问者,相亲交流平台相当于代理,被挑选的人相当于真正提供服务的对象。当挑选的人在向相亲交流平台要求得到被挑选的人的信息,实际上,提供信息的人不是平台,是被挑选的人。挑选的人在对被挑选的人打招呼,而不是对相亲交流平台打招呼。而且,相亲交流平台不只是为一个独特的挑选的人提供被挑选的人的信息,只要有忍看上被挑选的人,相亲交流平台就可以提供ta的信息。

 

CGLIB动态代理

此处仅突出CGLIB动态代理跟JDK动态代理在代理类上的区别:

import java.lang.reflect.Method;

import org.springframework.cglib.proxy.Enhancer;

import org.springframework.cglib.proxy.MethodInterceptor;

import org.springframework.cglib.proxy.MethodProxy;

public class HelloServiceCgLib implements MethodInterceptor{

         private Object target;

         public Object getInstance(Object target) {

                   this.target = target;

                   Enhancer enhancer = new Enhancer();

                   enhancer.setSuperclass(this.target.getClass());

                   //回调方法

                   enhancer.setCallback(this);

                   return enhancer.create();

         }

         @Override

         public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {

                   System.err.println("##########我是CGLIB的动态代理##########");

                   //反射方法前调用

                   System.err.println("我准备说hello");

                   Object returnObj = proxy.invokeSuper(obj, args);

                   //反射方法后调用

                   System.err.println("我说过hello了");

                   return returnObj;

         }

}

可以看到CGLIB的代理类是实现MethodInterceptor的代理方法。在mybatis中通常在延迟加载的时候才会用到CGLIB的动态代理。

(十二)mybatis之动态代理的更多相关文章

  1. Mybatis mapper动态代理的原理详解

    在开始动态代理的原理讲解以前,我们先看一下集成mybatis以后dao层不使用动态代理以及使用动态代理的两种实现方式,通过对比我们自己实现dao层接口以及mybatis动态代理可以更加直观的展现出my ...

  2. Spring事务Transactional和动态代理(二)-cglib动态代理

    系列文章索引: Spring事务Transactional和动态代理(一)-JDK代理实现 Spring事务Transactional和动态代理(二)-cglib动态代理 Spring事务Transa ...

  3. 浅谈Java代理二:Cglib动态代理-MethodInterceptor

    浅谈Java代理二:Cglib动态代理-MethodInterceptor CGLib动态代理特点: 使用CGLib实现动态代理,完全不受代理类必须实现接口的限制,而且CGLib底层采用ASM字节码生 ...

  4. aop学习总结二------使用cglib动态代理简单实现aop功能

    aop学习总结二------使用cglib动态代理简单实现aop功能 模拟业务需求: 1.拦截所有业务方法 2.判断用户是否有权限,有权限就允许用户执行业务方法,无权限不允许用户执行业务方法 (判断是 ...

  5. Java动态代理(二)——jdk动态代理

    一.什么是动态代理?代理类在程序运行时创建的代理方式被成为动态代理.动态代理的代理类并不是在Java代码中定义的,而是在运行时根据我们在Java代码中的“指示”动态生成的.相比于静态代理, 动态代理的 ...

  6. MyBatis学习(三)MyBatis基于动态代理方式的增删改查

    1.前言 上一期讲到MyBatis-Statement版本的增删改查.可以发现.这种代码写下来冗余的地方特别多.写一套没啥.如果涉及到多表多查询的时候就容易出现问题.故.官方推荐了一种方法.即MyBa ...

  7. 初看Mybatis 源码 (二) Java动态代理类

    先抛出一个问题,用过Mybatis的都知道,我们只需要定义一个Dao的接口,在里面写上一些CRUD相关操作,然后配置一下sql映射文件,就可以达到调用接口中的方法,然后执行sql语句的效果,为什么呢? ...

  8. Mybatis使用动态代理实现拦截器功能

    1.背景介绍 拦截器顾名思义为拦截某个功能的一个武器,在众多框架中均有“拦截器”.这个Plugin有什么用呢?或者说拦截器有什么用呢?可以想想拦截器是怎么实现的.Plugin用到了Java中很重要的一 ...

  9. JAVA框架 Spring 和Mybatis整合(动态代理)

    一.使用传统方式的dao的书写方式,不建议.目前采用的是动态代理的方式交给mybatis进行处理. 首先回顾下动态代理要求: 1)子配置文件的中,namespace需要是接口的全路径,id是接口的方法 ...

随机推荐

  1. IReport制作报表——日期时间显示格式

    转自:https://blog.csdn.net/linglinglu/article/details/9022679?utm_source=blogxgwz2 IReport工具在制作报表的时候,会 ...

  2. nodejs 循环的陷阱

    Node.js 的异步机制由事件和回调函数实现,一开始接触可能会感觉违反常规,但习惯 以后就会发现还是很简单的.然而这之中其实暗藏了不少陷阱,一个很容易遇到的问题就是 循环中的回调函数,初学者经常容易 ...

  3. idea创建vue项目,Terminal安装npm的淘宝镜像:'npm' 不是内部或外部命令,也不是可运行的程序 或批处理文件。

    原因: 安装node.js时,不是默认路径安装,环境变量找不到npm,需要改环境变量配置: 原下: 找到安装node.js的安装路径: 改后: 成功: npm i -g cnpm --registry ...

  4. 【Data Structure & Algorithm】 查找最小的k个元素

    查找最小的k个元素 题目:输入n个整数,输出其中最小的k个. 例如输入1, 2, 3, 4, 5, 6, 7和8这八个数字,则最小的4个数字为1, 2, 3和4. 分析:这道题最简单的思路是把输入的n ...

  5. httpd基础

    hpptd http服务器应用 http服务器程序 httpd apache nginx lighttpd 应用程序服务器 IIS .asp tomcat .jsp jetty 开源的servlet容 ...

  6. Laravel中的路由管理

    //路由中输出视图Route::get('/', function () { return view('welcome');}); //get路由请求Route::get('get',function ...

  7. ubuntu上安装与卸载deb文件(转载)

    转自:http://blog.csdn.net/nkguohao/article/details/8951082 版权声明:本文为博主原创文章,未经博主允许不得转载. 通过deb包安装软件: sudo ...

  8. Kinect SDK(1):读取彩色、深度、骨骼信息并用OpenCV显示

    Kinect SDK 读取彩色.深度.骨骼信息并用OpenCV显示 一.原理说明 对于原理相信大家都明白大致的情况,因此,在此只说比较特别的部分. 1.1 深度流数据: 深度数据流所提供的图像帧中,每 ...

  9. Visual Studio 调试卡顿的解决方法

    将不用的断点删除 关闭调试时打开的"IntelliXXX的窗口" 上面的第2条测试有效! https://social.msdn.microsoft.com/Forums/zh-C ...

  10. [Xcode 实际操作]九、实用进阶-(19)重写父类的绘图方法,使用图形上下文绘制自定义图形

    目录:[Swift]Xcode实际操作 本文将演示如何使用图形上下文,绘制自定义图形. 使用快捷键[Command]+[N]创建一个新的类文件. (在项目文件夹[DemoApp]上点击鼠标右键[New ...