Mybatis之MetaObject

news2024/11/29 9:00:21

在mybatis中,ResultSetHandler在收集JDBC返回的结果后需要转换成对应的Bean对象,其实映射的原理基本大家都能想到使用的时候java中的反射机制,但是在Mybatis中,提供了一个更加强大的对象,就是MetaObject,使用这个对象不仅对Bean中的属性赋值、取值都十分简单,同时还能对嵌套对象进行操作。

1、MetaObject

MetaObject 可以操作嵌套属性。

public class MetaObject {

  private final Object originalObject;
  private final ObjectWrapper objectWrapper;
  private final ObjectFactory objectFactory;
  private final ObjectWrapperFactory objectWrapperFactory;
  private final ReflectorFactory reflectorFactory;
	
  private MetaObject(Object object, ObjectFactory objectFactory, ObjectWrapperFactory objectWrapperFactory, 
  																ReflectorFactory reflectorFactory) {
	    this.originalObject = object;
	    this.objectFactory = objectFactory;
	    this.objectWrapperFactory = objectWrapperFactory;
	    this.reflectorFactory = reflectorFactory;
	
	    if (object instanceof ObjectWrapper) {
	      this.objectWrapper = (ObjectWrapper) object;
	    } else if (objectWrapperFactory.hasWrapperFor(object)) {
	      this.objectWrapper = objectWrapperFactory.getWrapperFor(this, object);
	    } else if (object instanceof Map) {
	      this.objectWrapper = new MapWrapper(this, (Map) object);
	    } else if (object instanceof Collection) {
	      this.objectWrapper = new CollectionWrapper(this, (Collection) object);
	    } else {
	      this.objectWrapper = new BeanWrapper(this, object);
	    }
  }

}
public static void main(String[] args) {

        MetaObjectBean bean = new MetaObjectBean();
        Configuration configuration = new Configuration();

        MetaObject metaObject = configuration.newMetaObject(bean);
        metaObject.setValue("username","李逵");

        metaObject.setValue("bean.address","山东");

        System.out.println(bean.getBean().getAddress());//山东

        List<MetaObjectChildBean> beanList = new ArrayList<>();
        beanList.add(bean.getBean());

        metaObject.setValue("beanList",beanList);
        System.out.println(((MetaObjectChildBean)metaObject.getValue("beanList[0]")).getAddress());//山东
    }

1.1、ObjectWrapper

在这里插入图片描述

1.1.1、BeanWrapper

BeanWapper 只能操本类的属性。

public class BeanWrapper extends BaseWrapper {

  private final Object object;
  private final MetaClass metaClass;

  public BeanWrapper(MetaObject metaObject, Object object) {
    super(metaObject);
    this.object = object;
    this.metaClass = MetaClass.forClass(object.getClass(), metaObject.getReflectorFactory());
  }
}

1.2、ObjectFactory

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);
  }

  private  <T> T instantiateClass(Class<T> type, List<Class<?>> constructorArgTypes, List<Object> constructorArgs) {
    try {
      Constructor<T> constructor;
      if (constructorArgTypes == null || constructorArgs == null) {
        constructor = type.getDeclaredConstructor();
        return constructor.newInstance();
      }
      constructor = type.getDeclaredConstructor(constructorArgTypes.toArray(new Class[0]));
      return constructor.newInstance(constructorArgs.toArray(new Object[0]));
    } 
  }

  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;
  }
}

1.3、ObjectWrapperFactory

public class DefaultObjectWrapperFactory implements ObjectWrapperFactory {

  @Override
  public boolean hasWrapperFor(Object object) {
    return false;
  }

  @Override
  public ObjectWrapper getWrapperFor(MetaObject metaObject, Object object) {
    throw new ReflectionException("The DefaultObjectWrapperFactory should never be called to provide an 
    			ObjectWrapper.");
  }
}

1.4、ReflectorFactory

public class DefaultReflectorFactory implements ReflectorFactory {
  private boolean classCacheEnabled = true;
  private final ConcurrentMap<Class<?>, Reflector> reflectorMap = new ConcurrentHashMap<>();

  public DefaultReflectorFactory() {
  }

  @Override
  public boolean isClassCacheEnabled() {
    return classCacheEnabled;
  }

  @Override
  public void setClassCacheEnabled(boolean classCacheEnabled) {
    this.classCacheEnabled = classCacheEnabled;
  }

  @Override
  public Reflector findForClass(Class<?> type) {
    if (classCacheEnabled) {
      // synchronized (type) removed see issue #461
      return reflectorMap.computeIfAbsent(type, Reflector::new);
    } else {
      return new Reflector(type);
    }
  }

}

2、MetaClass

可以操作嵌套的set、get方法.

private MetaClass(Class<?> type, ReflectorFactory reflectorFactory) {
  this.reflectorFactory = reflectorFactory;
  this.reflector = reflectorFactory.findForClass(type);
}

2.1、属性分词器之PropertyTokenizer

当需要集合中某个元素的某个属性的时候,表达式需要这样写:list[0].user.name,这个时候就需要一个比较重要的组件去判断是否直接获取属性,还是递归生成对象了,它就是PropertyTokenizer分词器。

public static void main(String[] args) {

    Reflector reflector = new Reflector(Tokenizer.class);
    token(reflector.findPropertyName("username"));
    token("address.number");
    token("address[1].number");
}

private static void token(String username) {
    PropertyTokenizer tokenizer = new PropertyTokenizer(username);
    String children = tokenizer.getChildren();
    String index = tokenizer.getIndex();
    String indexedName = tokenizer.getIndexedName();
    String name = tokenizer.getName();
    log.error("children:{} index:{} indexedName:{} name:{}",children,index,indexedName,name);
}
children:null index:null indexedName:username name:username
children:number index:null indexedName:address name:address
children:number index:1 indexedName:address[1] name:address

3、Reflector

public class Reflector {
	  private final Class<?> type;
	  private final String[] readablePropertyNames;
	  private final String[] writablePropertyNames;
	  private final Map<String, Invoker> setMethods = new HashMap<>();
	  private final Map<String, Invoker> getMethods = new HashMap<>();
	  private final Map<String, Class<?>> setTypes = new HashMap<>();
	  private final Map<String, Class<?>> getTypes = new HashMap<>();
	  private Constructor<?> defaultConstructor;
	  
	public Reflector(Class<?> clazz) {
		  type = clazz;
		  addDefaultConstructor(clazz);
		  addGetMethods(clazz);
		  addSetMethods(clazz);
		  addFields(clazz);
		  readablePropertyNames = getMethods.keySet().toArray(new String[0]);
		  writablePropertyNames = setMethods.keySet().toArray(new String[0]);
		  for (String propName : readablePropertyNames) {
		    caseInsensitivePropertyMap.put(propName.toUpperCase(Locale.ENGLISH), propName);
		  }
		  for (String propName : writablePropertyNames) {
		    caseInsensitivePropertyMap.put(propName.toUpperCase(Locale.ENGLISH), propName);
		  }
	}
}

该类的作用是通过反射将类clazz的相关属性提前都初始化完毕,并缓存处理。当需要类相关属性时无需手动每次反射处理,只需通过Reflector公开方法直接获取即可。

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

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

相关文章

Python - 面向对象编程 - 实例方法、静态方法、类方法

实例方法 在类中定义的方法默认都是实例方法&#xff0c;前面几篇文章已经大量使用到实例方法 实例方法栗子 class PoloBlog:def __init__(self, name, age):print("自动调用构造方法")self.name nameself.age agedef test(self):print("一个实例方法&…

阿里P8写出的《深入理解Java虚拟机》最新版,轻松学会JVM底层

前言 Java是目前用户最多、使用范围最广的软件开发技术&#xff0c;Java的技术体系主要由支撑Java程序运行的虚拟机、提供各开发领域接口支持的Java类库、Java编程语言及许许多多的第三E方Java框架(如Spring、 MyBatis等) 构成。在国内&#xff0c;有关Java类库API、Java语言语…

能源汽车下乡充电桩建设优化建议及解决方案

安科瑞 耿敏花 摘 要&#xff1a;5月5日&#xff0c;国务院总理李强主持召开国务院常务会议&#xff0c;审议通过关于加快发展先进制造业集群的意见&#xff0c;部署加快建设充电基础设施&#xff0c;更好支持新能源汽车下乡和乡村振兴。 会议指出&#xff0c;农村新能源汽车市…

JavaScript 原型和原型链

文章目录 JavaScript 原型和原型链概述new操作符的流程原型对象、构造函数、实例关系原型链原型链查找hasOwnPropertyin JavaScript 原型和原型链 概述 任何函数都有prototype属性&#xff0c;prototype是原型的意思。 prototype的属性值是个对象&#xff0c;它默认拥有cons…

Java内存模型(Java Memory Mode,JMM)

并发编程模型的两个关键问题 线程之间如何通信及线程之间如何同步。 线程之间如何通信&#xff1a;共享内存&#xff0c;消息传递线程之间如何同步通信是指线程之间以何种机制来 交换信息同步是指程序中用于控制不同线程间 操作发生相对顺序 的机制在共享内存的并发模型里&a…

网络安全大厂面试题汇总

注&#xff1a;本套面试题&#xff0c;已整理成pdf文档&#xff0c;但内容还在持续更新中&#xff0c;因为无论如何都不可能覆盖所有的面试问题&#xff0c;更多的还是希望由点达面&#xff0c;查漏补缺。 ​ 某大厂一面整理 1、防范常见的 Web 攻击 2、重要协议分布层 3、ar…

c++使用yaml -基于windows10

参考&#xff1a;Windows10下使用VS2017编译和使用yaml-cpp库_雪域迷影的博客-CSDN博客 1. 下载yaml-cpp 建议在github下载其最新的官方版本&#xff0c;不要在其他平台下载该工具软件&#xff0c;下载地址如下&#xff08;其中的一个版本&#xff09;&#xff1a; Release …

理解深度可分离卷积

1、常规卷积 常规卷积中&#xff0c;连接的上一层一般具有多个通道&#xff08;这里假设为n个通道&#xff09;&#xff0c;因此在做卷积时&#xff0c;一个滤波器&#xff08;filter&#xff09;必须具有n个卷积核&#xff08;kernel&#xff09;来与之对应。一个滤波器完成一…

Android自定义键盘(KeyboardView)

目录 1.场景&#xff1a;2.想法&#xff1a;3.开始实现&#xff1a;《一》 在res包下创建xml目录,Keyboard标签来定义键盘布局&#xff1a;《二》创建IKeyboardView类并继承KeyboardView类,设置键盘布局&#xff08;数字和字母&#xff09;《三》 处理自定义键盘按键的点击事件…

C++11:列表初始化、新增关键字和新增的默认成员函数

目录 一. 列表初始化 1.1 {}列表初始化的方法 1.2 列表初始化实现的原理 二. C11新增关键字 2.1 decltype -- 自动类型推断 2.2 nullptr -- 空指针 2.3 default -- 强制生成默认成员函数 2.4 delete -- 禁止生成默认成员函数 2.5 final -- 禁止类被继承/禁止虚函数被重…

自动化测试之PageObject设计模式

译文如下&#xff1a; PageObject 当您针对网页编写测试时&#xff0c;您需要参考该网页中的元素以单击链接并确定显示的内容。但是&#xff0c;如果您编写直接操作 HTML 元素的测试用例&#xff0c;则您的测试将无法应对 UI 中的频繁修改。PageObject对应于一个HTML网页、页…

5月26号软件资讯更新合集......

Windows Terminal 1.18 新功能预览&#xff1a;标签拖拽、上下文菜单... Windows Terminal 预览版已更新到 1.18 版本&#xff0c;带来多项实用内容&#xff0c;一起来看看这个版本的新东西&#xff1a; 标签撕裂&#xff08;拖拽功能&#xff09; Windows Terminal 已支持对…

安装Redis6

安装Redis 安装redis依赖 redis是基于C语言编写的&#xff0c;因此首选需要安装redis所需的gcc依赖 yum install -y gcc tcl 上传安装包并解压 我安装的是redis-6.2.6版本&#xff0c;并且放到了/usr/local/src目录下 - 进入/usr/local/src目录shellcd /usr/local/src解压 tar …

全球最受欢迎低代码平台排行榜出炉

低代码平台正在成为寻求快速有效地构建应用程序的企业的首选解决方案。这些平台减少了编码要求&#xff0c;使企业能够在降低成本的同时更快地完成应用程序开发项目。在本文中&#xff0c;将探索全球受欢迎的低代码平台排行榜。 该排名使用的标准包括易用性、成本效益、集成性、…

很后悔,才发现这个API管理神器

想必大家都注意到了&#xff0c;近半年国产API管理工具火了起来。这说明两个问题&#xff0c;第一&#xff0c;API管理的重要性被越来越多的开发者认识到了&#xff0c;研发团队对API管理的需求也越来越强了。第二&#xff0c;说明国产软件真是越来越厉害了&#xff0c;大家确实…

《微服务实战》 第十八章 Redis查看配置文件和数据类型

前言 本章节讲解如何查看、修改Redis配置&#xff0c;介绍Redis类型。 1、查看配置 config get 配置名称 2、修改配置项 config set 配置项名称 配置项值 2.1、配置项说明 配置项参数说明daemonizeno/yes默认为 no&#xff0c;表示 Redis 不是以守护进程的方式运行&#xff…

论C站如何获得铁粉?过来人给出几点建议

哈喽&#xff0c;我是bug菌&#xff0c;一名想走&#x1f463;出大山改变命运的程序猿。周五啦&#xff0c;刚肝完需求的我&#xff0c;闲暇之时逛C站热榜&#xff0c;偶然刷到一条看到官方抛出的话题&#xff1a;"在C站如何获得铁粉&#xff1f;"&#xff0c;我寻思…

操作系统第五章——输入输出管理(上)

提示&#xff1a;初入红尘&#xff0c;不知人间疾苦&#xff0c;蓦然回首&#xff0c;已是苦中之人&#xff0c;这杯中酒三分&#xff0c;这酒中悲七分。关关难过关关过&#xff0c;夜夜难熬夜夜熬&#xff0c;愿这人世间所有爱恨情仇皆溶于酒&#xff0c;且将这红尘做酒&#…

MP4如何让去水印?python带你实现~

前言 嗨喽&#xff0c;大家好呀~这里是爱看美女的茜茜呐 开发环境: 解释器版本: python 3.8 代码编辑器: pycharm 2021.2 模块使用: 内置模块&#xff08;无需安装&#xff09; os —> python系统编程的操作模块,提供了非常丰富的功能去处理文件和目录 sys —> 是与…

我是00后,我卷一点怎么了?

前段时间去面试了一个公司&#xff0c;成功拿到了offer&#xff0c;薪资也从12k涨到了18k&#xff0c;对于工作都还没两年的我来说&#xff0c;还是比较满意的&#xff0c;毕竟一些工作3、4年的可能还没我高。 我可能就是大家说的卷王&#xff0c;感觉自己年轻&#xff0c;所以…