通过之前的介绍,我们了解了几个组件的生命周期。

它也是我们重要装备之一。

今天我们需要搞一件更加强的装备,叫做反射和动态代理。

如果没有这件装备的话,显然后面的源码boss是打不动的。

顺便说一下,下面可能会提到一些名词简写,都在之前出现过,所以如果看不明白可以看一下之前的咯。

废话少说,╭(′▽`)╯

什么是代理

首先我们要明白代理是什么样一个概念。

不要想这个概念有多复杂,其实很简单的。代理从中文角度很好理解嘛。它是一个动词吧,举个实际的例子,你代理我,帮我抄作业。

我--->你--->作业

你代理我,相当于我让你做,或者你帮我做。

最终抄作业这个事情呢是由你来完成的。嘻嘻。

注意这个例子的两个关键:

1、我没有直接操作真实的最终的对象。(我没有抄作业)

2、你可以提供额外的服务,比如告诉我作业抄好了,或者明天有什么作业。

那么认真的讲:

通过代理模式,访问者没有办法操作真实的对象,只能通过代理对象去操作真实的对象。代理对象可以提供额外的服务,比操作真实对象更多的服务。

什么是反射

是不是很多人当提到这个概念有下面几个感觉:1、反射学过没用过。2、反射很难,但是没啥用。3、反射很危险,没事情不要乱用。

如果你有这么几个概念的话,证明你已经对反射有一些学习和感觉了(滑稽脸)。

如果没有的话,我建议呢,还是去好好找个大神的博客或者视频学习一下反射。因为不简单,听我一两句可能不一定明白。

那我们这里用到反射,需要知道这几个点。

1、在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法。

2、对于任意一个对象,都能够调用它的任意方法和属性;生成动态代理。

3、通过一些特殊的方法得到对象,calss.forname等

知道这些就可以了。

如果你愿意呢,可以看一下我以前写的关于反射的几个小例子,希望能帮到你:http://www.cnblogs.com/linkstar/p/5744444.html

什么是动态代理

动态代理其实上面在说代理的时候以及提到了。其实就是动态的代理嘛。

动态代理这里介绍两种:一种使用JDK的反射机制提供的代理,另一种是CGLIB代理。

我们一种种来讲。

JDK的动态代理。

如何实现呢?很简单,完成两个类就可以了。

1、服务类,也就是真正提供服务的人。

2、代理类,代理做事情的人。

对于JDK的动态代理的服务类来说需要额外一个接口开放给代理类哦。

下面是代码

接口

public interface DemoService {
public void demo();
}

实现类

public class DemoServiceImpl implements DemoService {
public void demo() {
System.out.println("Test------------");
}
}

代理类

public class DemoServiceProxy implements InvocationHandler {

    private Object target;

    public Object bind(Object target){
this.target = target;
//利用反射将这个类DemoServiceProxy作为传入的目标类的代理类
return Proxy.newProxyInstance(target.getClass().getClassLoader(), target.getClass().getInterfaces(), this);
} //代理方法
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
System.out.println("method : " + method.getName());
return method.invoke(target, args);
}
}

测试类

public class DemoServiceTest {
public static void main(String[] args) {
DemoServiceProxy demoServiceHanler = new DemoServiceProxy();
DemoService proxy = (DemoService) demoServiceHanler.bind(new DemoServiceImpl());
proxy.demo();
}
}

最后打印结果

method : demo
Test------------

这里的这几个类的作用和功能要好好理解哦,因为后面读源码的时候需要用到。

然后是CGLIB代理

相对来说就没有那么复杂了,只需要一个类就可以了,不需要接口。

public class DemoServiceCglib implements MethodInterceptor {

    private Object target;

    public Object getInstance(Object target){
this.target = target;
Enhancer enhancer = new Enhancer();
enhancer.setSuperclass(target.getClass());
enhancer.setCallback(this);
return enhancer.create();
} public Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {
System.out.println("method : " + method.getName());
return proxy.invokeSuper(obj, args);
}
}

测试类修改

public class DemoServiceTest {
public static void main(String[] args) {
/*DemoServiceProxy demoServiceHanler = new DemoServiceProxy();
DemoService proxy = (DemoService) demoServiceHanler.bind(new DemoServiceImpl());
proxy.demo();*/
DemoServiceCglib demoServiceCglib = new DemoServiceCglib();
DemoService proxy = (DemoService) demoServiceCglib.getInstance(new DemoServiceImpl());
proxy.demo();
}
}

测试结果相同。

这里就不深究两种动态代理的各种参数啊,方法啊。

我们需要知道的是,在被动态代理之后,当我们调用原来的方法时候一定调用一个叫做invoke的方法。

也就是在调用的前后我们能做很多事情了。

上述在mybatis的源码中会出现,很重要,望理解。

至此,装备都有了,之后就要开始进入难点了。

MyBatis源码解析【4】反射和动态代理的更多相关文章

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

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

  2. MyBatis源码解析之数据源(含数据库连接池简析)

    一.概述: 常见的数据源组件都实现了javax.sql.DataSource接口: MyBatis不但要能集成第三方的数据源组件,自身也提供了数据源的实现: 一般情况下,数据源的初始化过程参数较多,比 ...

  3. Mybatis源码解析(四) —— SqlSession是如何实现数据库操作的?

    Mybatis源码解析(四) -- SqlSession是如何实现数据库操作的?   如果拿一次数据库请求操作做比喻,那么前面3篇文章就是在做请求准备,真正执行操作的是本篇文章要讲述的内容.正如标题一 ...

  4. mybatis源码-解析配置文件(四-1)之配置文件Mapper解析(cache)

    目录 1. 简介 2. 解析 3 StrictMap 3.1 区别HashMap:键必须为String 3.2 区别HashMap:多了成员变量 name 3.3 区别HashMap:key 的处理多 ...

  5. mybatis源码-解析配置文件(四)之配置文件Mapper解析

    在 mybatis源码-解析配置文件(三)之配置文件Configuration解析 中, 讲解了 Configuration 是如何解析的. 其中, mappers作为configuration节点的 ...

  6. mybatis源码-解析配置文件(三)之配置文件Configuration解析

    目录 1. 简介 1.1 系列内容 1.2 适合对象 1.3 本文内容 2. 配置文件 2.1 mysql.properties 2.2 mybatis-config.xml 3. Configura ...

  7. Mybatis源码解析,一步一步从浅入深(四):将configuration.xml的解析到Configuration对象实例

    在Mybatis源码解析,一步一步从浅入深(二):按步骤解析源码中我们看到了XMLConfigBuilder(xml配置解析器)的实例化.而且这个实例化过程在文章:Mybatis源码解析,一步一步从浅 ...

  8. Mybatis源码解析,一步一步从浅入深(五):mapper节点的解析

    在上一篇文章Mybatis源码解析,一步一步从浅入深(四):将configuration.xml的解析到Configuration对象实例中我们谈到了properties,settings,envir ...

  9. Mybatis源码解析,一步一步从浅入深(六):映射代理类的获取

    在文章:Mybatis源码解析,一步一步从浅入深(二):按步骤解析源码中我们提到了两个问题: 1,为什么在以前的代码流程中从来没有addMapper,而这里却有getMapper? 2,UserDao ...

  10. Mybatis源码解析(三) —— Mapper代理类的生成

    Mybatis源码解析(三) -- Mapper代理类的生成   在本系列第一篇文章已经讲述过在Mybatis-Spring项目中,是通过 MapperFactoryBean 的 getObject( ...

随机推荐

  1. js事件小结

    首先事件绑定分为2种方法 一种为"DOM0级"方法,这里我理解为事件指定 var oDiv = document.getElementById("div1"); ...

  2. [原创]Nexus5 内核编译烧录过程记录

    参考Android系统源代码情况分析第二章进行实践,为了提高效率,也为了增加实践机会,使用Nexus5进行内核编译.需要说明的是,Android源代码工程默认是不包含它所使用的Linux内核源码,如果 ...

  3. iStat for mac

    iStat for mac 电脑硬件信息检测软件,安装完成后它位于"系统偏好设定"的应用程序面板,让您从选单列监测系统的各项丰富资讯,又不会占用使用者太大的桌面空间,提供的信息包括 ...

  4. 全局精确流量调度新思路-HttpDNS服务详解

    但凡使用域名来给用户提供服务的互联网企业,都或多或少地无法避免在有中国特色的互联网环境中遭遇到各种域名被缓存.用户跨网访问缓慢等问题.那么对于腾讯这样的域名数量在10万级别的互联网公司来讲,域名解析异 ...

  5. 《HelloGitHub》第 14 期

    公告 欢迎通过在 GitHub 上新建 issues 方式推荐项目,我真心希望读者可以在 HelloGItHub,找到真正的编程乐趣! <HelloGitHub>第 14 期 兴趣是最好的 ...

  6. 两种进入容器的方法 - 每天5分钟玩转 Docker 容器技术(23)

    我们经常需要进到容器里去做一些工作,比如查看日志.调试.启动其他进程等.有两种方法进入容器:attach 和 exec. docker attach 通过 docker attach 可以 attac ...

  7. php中的数组遍历的几种方式

    [(重点)数组循环遍历的四种方式]   1.使用for循环遍历数组     conut($arr);用于统计数组元素的个数.     for循环只能用于遍历,纯索引数组!!!!     如果存在关联数 ...

  8. ASP.NET MVC5(一):ASP.NET MVC概览

    ASP.NET MVC概览 ASP.NET MVC是一种构建Web应用程序的框架,它将一般的MVC(Model-View-Controller)模式应用于ASP.NET框架. 1.ASP.NET MV ...

  9. JavaScript 循环性能比较

    有一句话叫做 没有什么事是一个循环解决不了的,如果真有,那就再来一个循环. 循环的种类有很多 正向for循环,逆向for循环,while循环,for-in循环(理论上性能最差),for-each循环, ...

  10. javacv开发详解之1:调用本机摄像头视频(建议使用javaCV1.3版本)

    javaCV系列文章: javacv开发详解之1:调用本机摄像头视频 javaCV开发详解之2:推流器实现,推本地摄像头视频到流媒体服务器以及摄像头录制视频功能实现(基于javaCV-FFMPEG.j ...