2019独角兽企业重金招聘Python工程师标准>>>

TypeParameterResolver:

TypeParameterResolver的功能是:当存在复杂的继承关系以及泛型定义时, TypeParameterResolver 可以帮助我们解析字段、方法参数或方法返回值的类型。TypeParameterResolver 是在Refelctor中的addGetMethod方法中调用的,目的是获取方法的返回值类型。在Refelctor中的addSetMethod方法中调用的,目的是获取方法的参数类型。

private void addGetMethod(String name, Method method) {//检查属性名是否合法,检查条件方法名不以$开头,不等于serialVersionUID  不等于class
  if (isValidPropertyName(name)) {getMethods.put(name, new MethodInvoker(method));Type returnType = TypeParameterResolver.resolveReturnType(method, type);getTypes.put(name, typeToClass(returnType));}
}
private void addSetMethod(String name, Method method) {if (isValidPropertyName(name)) {setMethods.put(name, new MethodInvoker(method));Type[] paramTypes = TypeParameterResolver.resolveParamTypes(method, type);setTypes.put(name, typeToClass(paramTypes[0]));}
}

下面来看TypeParameterResolver具体实现:首先看resolveFieldType(),resolveReturnType(),resolveParamTypes()
此文已resolveFieldType为例讲解。

resolveFileType 第一步获取字段的声明类型,第二步 获取字段定义所在的类的Class对象。第三步resolveType是获取字段的类型

public static Type resolveFieldType(Field field, Type srcType) {//获取字段的声明类型
  Type fieldType = field.getGenericType();//获取字段定义所在的类的Class 对象
  Class<?> declaringClass = field.getDeclaringClass();return resolveType(fieldType, srcType, declaringClass);
}

public static Type resolveReturnType(Method method, Type srcType) {Type returnType = method.getGenericReturnType();Class<?> declaringClass = method.getDeclaringClass();return resolveType(returnType, srcType, declaringClass);
}

public static Type[] resolveParamTypes(Method method, Type srcType) {Type[] paramTypes = method.getGenericParameterTypes();Class<?> declaringClass = method.getDeclaringClass();Type[] result = new Type[paramTypes.length];for (int i = 0; i < paramTypes.length; i++) {result[i] = resolveType(paramTypes[i], srcType, declaringClass);}return result;
}

此处需要先介绍一下Type接口,Type是所有类型的父接口,它有四个子类和一个实现类。

Class 比较常见,它表示的是原始类型。Class 类的对象表示NM 中的一个类或接口,每个Java 类在NM 里都表现为一个Class 对象。在程序中可以通过“类名.class ”、“对象.getC!ass()”或是Class 对象,所有元素类型相同且维数相同的数组都共享同一个Class 对象。
ParameterizedType 表示的是参数化类型,例如List<String> 、Map<Integer,String>、Service<U ser>这种带有泛型的类型。
Type Variable 表示的是类型变量,它用来反映在NM 编译该泛型前的信息。例如List<T>中的T 就是类型变量,它在编译时需被转换为一个具体的类型后才能正常使用。

GenericArrayType 表示的是数组类型且组成元素是ParameterizedType 或Type Variable .例如List<String>[]或T [] 。该接口只有Type getGenericComponentType () 一个方法,它返回数组的组成元素。
WildcardType 表示的是通配符泛型,例如? extends Number 和? super Integer 。

现在看 resolveType(Type type, Type srcType, Class<?> declaringClass) 源码:主要是根据字段类型来匹配是属于哪个类型的,然后返回

private static Type resolveType(Type type, Type srcType, Class<?> declaringClass) {if (type instanceof TypeVariable) {return resolveTypeVar((TypeVariable<?>) type, srcType, declaringClass);} else if (type instanceof ParameterizedType) {return resolveParameterizedType((ParameterizedType) type, srcType, declaringClass);} else if (type instanceof GenericArrayType) {return resolveGenericArrayType((GenericArrayType) type, srcType, declaringClass);} else {return type;}
}

ObjectFactory  

   ObjectFactory 主要功能是:根据指定的参数列表查找构造函数,并实例化对象。

My Batis 中有很多模块会使用到ObjectFactory 接口,该接口提供了多个create()方法的重载,通过这些create()方法可以创建指定类型的对象。

public interface ObjectFactory {/**
   * Sets configuration properties. 设置配置信息
   * @param properties configuration properties
   */
  void setProperties(Properties properties);/**
   * Creates a new object with default constructor. 通过无参构造器创建对象
   * @param type Object type
   * @return
   */
  <T> T create(Class<T> type);/**
   * Creates a new object with the specified constructor and params.根据参数列表,从指定类型中选择合适的构造器创建对象
   * @param type Object type
   * @param constructorArgTypes Constructor argument types
   * @param constructorArgs Constructor argument values
   * @return
   */
  <T> T create(Class<T> type, List<Class<?>> constructorArgTypes, List<Object> constructorArgs);/**检测指定类型是否为集合类型,主要处理java.util.Collection及其子类。
   * Returns true if this object can have a set of other objects.
   * It's main purpose is to support non-java.util.Collection objects like Scala collections.
   *
   * @param type Object type
   * @return whether it is a collection or not
   * @since 3.1.0
   */
  <T> boolean isCollection(Class<T> type);}

ObjectFactory只有一个默认实现DafaultObjectFactory,主要看instantiateClass()方法,实现的功能就是 根据指定的参数列表查找构造函数,并实例化对象。

public class DefaultObjectFactory implements ObjectFactory, Serializable {private static final long serialVersionUID = -8855120656740914948L;@Override
  public <T> T create(Class<T> type) {return create(type, null, null);}@SuppressWarnings("unchecked")@Override
  public <T> T create(Class<T> type, List<Class<?>> constructorArgTypes, List<Object> constructorArgs) {Class<?> classToCreate = resolveInterface(type);// we know types are assignable
    return (T) instantiateClass(classToCreate, constructorArgTypes, constructorArgs);}@Override
  public void setProperties(Properties properties) {// no props for default
  }private  <T> T instantiateClass(Class<T> type, List<Class<?>> constructorArgTypes, List<Object> constructorArgs) {try {//声明构造方法
      Constructor<T> constructor;//如果参数列表为null,通过无参构造函数创建对象。
      if (constructorArgTypes == null || constructorArgs == null) {constructor = type.getDeclaredConstructor();if (!constructor.isAccessible()) {constructor.setAccessible(true);}return constructor.newInstance();}//根据指定的参数列表查找构造函数,并实例化对象。
      constructor = type.getDeclaredConstructor(constructorArgTypes.toArray(new Class[constructorArgTypes.size()]));if (!constructor.isAccessible()) {constructor.setAccessible(true);}return constructor.newInstance(constructorArgs.toArray(new Object[constructorArgs.size()]));} catch (Exception e) {StringBuilder argTypes = new StringBuilder();if (constructorArgTypes != null && !constructorArgTypes.isEmpty()) {for (Class<?> argType : constructorArgTypes) {argTypes.append(argType.getSimpleName());argTypes.append(",");}argTypes.deleteCharAt(argTypes.length() - 1); // remove trailing ,
      }StringBuilder argValues = new StringBuilder();if (constructorArgs != null && !constructorArgs.isEmpty()) {for (Object argValue : constructorArgs) {argValues.append(String.valueOf(argValue));argValues.append(",");}argValues.deleteCharAt(argValues.length() - 1); // remove trailing ,
      }throw new ReflectionException("Error instantiating " + type + " with invalid types (" + argTypes + ") or values (" + argValues + "). Cause: " + e, e);}}protected Class<?> resolveInterface(Class<?> type) {Class<?> classToCreate;if (type == List.class || type == Collection.class || type == Iterable.class) {classToCreate = ArrayList.class;} else if (type == Map.class) {classToCreate = HashMap.class;} else if (type == SortedSet.class) { // issue #510 Collections Support
      classToCreate = TreeSet.class;} else if (type == Set.class) {classToCreate = HashSet.class;} else {classToCreate = type;}return classToCreate;}@Override
  public <T> boolean isCollection(Class<T> type) {return Collection.class.isAssignableFrom(type);}}

转载于:https://my.oschina.net/u/3905482/blog/2249537

MyBatis源码分析-2-基础支持层-反射模块-TypeParameterResolver/ObjectFactory相关推荐

  1. MyBatis 源码分析 - SQL 的执行过程

    本文速览 本篇文章较为详细的介绍了 MyBatis 执行 SQL 的过程.该过程本身比较复杂,牵涉到的技术点比较多.包括但不限于 Mapper 接口代理类的生成.接口方法的解析.SQL 语句的解析.运 ...

  2. MyBatis 源码分析 - 映射文件解析过程

    1.简介 在上一篇文章中,我详细分析了 MyBatis 配置文件的解析过程.由于上一篇文章的篇幅比较大,加之映射文件解析过程也比较复杂的原因.所以我将映射文件解析过程的分析内容从上一篇文章中抽取出来, ...

  3. MyBatis 源码分析系列文章导读

    1.本文速览 本篇文章是我为接下来的 MyBatis 源码分析系列文章写的一个导读文章.本篇文章从 MyBatis 是什么(what),为什么要使用(why),以及如何使用(how)等三个角度进行了说 ...

  4. Mybatis源码分析第一天------Mybatis实用篇

    Mybatis源码分析第一天------Mybatis实用篇 一切最基本的操作就是参考官方文档:https://mybatis.org/mybatis-3/zh/configuration.html ...

  5. MyBatis 源码分析系列文章导读 1

    1.本文速览 本篇文章是我为接下来的 MyBatis 源码分析系列文章写的一个导读文章.本篇文章从 MyBatis 是什么(what),为什么要使用(why),以及如何使用(how)等三个角度进行了说 ...

  6. MyBatis源码分析(一)MyBatis整体架构分析

    文章目录 系列文章索引 一.为什么要用MyBatis 1.原始JDBC的痛点 2.Hibernate 和 JPA 3.MyBatis的特点 4.MyBatis整体架构 5.MyBatis主要组件及其相 ...

  7. MyBatis 源码分析 - 配置文件解析过程

    文章目录 * 本文速览 1.简介 2.配置文件解析过程分析 2.1 配置文件解析入口 2.2 解析 properties 配置 2.3 解析 settings 配置 2.3.1 settings 节点 ...

  8. 十年老架构师神级推荐,MyBatis源码分析,再也不用为源码担忧了

    十年老架构师神级推荐,MyBatis源码分析,再也不用为源码担忧了 前言 MyBatis是一个优秀的持久层ORM框架,它对jdbc的操作数据库的过程进行封装,使开发者只需要关注SQL 本身,而不需要花 ...

  9. mybatis源码分析之事务管理器

    2019独角兽企业重金招聘Python工程师标准>>> 上一篇:mybatis源码分析之Configuration 主要分析了构建SqlSessionFactory的过程中配置文件的 ...

最新文章

  1. 亿条数据读取工具_仅需1秒!搞定100万行数据:超强Python数据分析利器
  2. Android应用中网络请求库Volley的使用
  3. 2019小程序没必要做了_2019微信小程序的发展前景怎么样?有必要开发微信小程序吗?...
  4. 剑指offer之56-60题解
  5. 机器学习——决策树的三种学习方法
  6. SDH光传输设备是什么?SDH设备特点介绍
  7. java excel 删除单元格格式_Java 去除excel表格内容中的空格
  8. 两年半php,两年半,我收获了什么?
  9. loj #6122. 「网络流 24 题」航空路线问题
  10. 数据库系统工程师考试试题分类精解pdf
  11. thinkpade450装内存条_thinkpad e450可以加内存条吗
  12. Windows server2012R2 企业内部搭建虚拟专用网络服务
  13. AV1代码学习:解码端aomdec.c的main函数
  14. Mysql为什么使用B+树(一)之红黑树简述
  15. iPad做Windows扩展屏
  16. ddr布线 pads_[转载]Pads Router布线技巧,等长,蛇线,差分
  17. Java集合框架笔记,足够深,足够全面
  18. sio.savemat得到空struct解决方法
  19. c++实训 选猴子当大王
  20. AutoAugment 学习

热门文章

  1. 【Hadoop Summit Tokyo 2016】企业数据分类和治理
  2. LINUX下的tty,console与串口分析
  3. 网络编辑日常工作的内容有哪些?
  4. (转)Solaris 10 的基本操作
  5. 商战特训营 竞合赛场显英才
  6. sql语句练习(二):Demand
  7. $\mathfrak {reputation}$
  8. [源码和文档分享]基于Java的带GUI界面猜数字游戏
  9. 学习笔记26_MVC前台强类型参数
  10. Java中Comparable和Comparator接口区别分析