IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 大数据 -> MyBatis里面为什么老是实现HashMap重写get方法呢1 -> 正文阅读

[大数据]MyBatis里面为什么老是实现HashMap重写get方法呢1

作者:recommend-item-box type_blog clearfix

一、前言

DynamicContext(在scripting.xmltags包)里面封装了静态内部类ContextMap,并且ContextMap作为DynamicContext的成员变量bindings,用于封装用户入参parameterObject的。在《成员变量的作用》一节有讲到,bindings成员变量这里装载数据作用,关于存数据这里DynamicContext封装了bind()方法;

二、MyBatis的ContextMap实现HashMap重写get方法

MetaObject封装用户入参parameterObject

MetaObject metaObject = configuration.newMetaObject(parameterObject);
ContextMap封装MetaObject,间接就是封装了用户入参parameterObject
bindings = new ContextMap(metaObject, existsTypeHandler);

为了方便理解,ContextMap也可以理解为一种数据库的作用只是没有持久化,下面就把ContextMap当作数据库看待。而用ContextMap继承HashMap,是因为HashMap的get方法没有满足需求。因为value可能不在HashMap里面,可能存在parameterMetaObjec里面,也可能NULL

1、ContextMap的get方法(完整源码请看下面DynamicContext完整源码)

   // 这里好比分表,HashMap是一个表,parameterMetaObject另一个表,一个个获取,也可能返回空。可能比喻不是很恰当
    @Override
    public Object get(Object key) {
      String strKey = (String) key;
      // 判断数据库是否有了该数据
      if (super.containsKey(strKey)) {
        return super.get(strKey);
      }

      // 数据库没有数据,判断parameterMetaObject是否空
      if (parameterMetaObject == null) {
        return null;
      }
      
      // 上面parameterMetaObject非空,这里调用方法就不会出现NULL异常了
      if (fallbackParameterObject && !parameterMetaObject.hasGetter(strKey)) {
        return parameterMetaObject.getOriginalObject();
      } else {
        // issue #61 do not modify the context when reading
        return parameterMetaObject.getValue(strKey);
      }
    }

2、DynamicContext完整源码

/**
 * 记录 动态SQL语句解析结果的 容器
 * @author Clinton Begin
 */
public class DynamicContext {

  public static final String PARAMETER_OBJECT_KEY = "_parameter";
  public static final String DATABASE_ID_KEY = "_databaseId";

  static {
    OgnlRuntime.setPropertyAccessor(ContextMap.class, new ContextAccessor());
  }

  private final ContextMap bindings; // 参数上下文,内部类成员变量,实现HashMap并重写get()方法
  // 在SqlNode解析动态SQL时,会将解析后的SQL语句片段添加到该属性中保存,最终拼凑出一条完整的SQL语句
  private final StringJoiner sqlBuilder = new StringJoiner(" ");
  private int uniqueNumber = 0;

  /**
   * 公有构造方法 职责是:初始化bindings集合
   * @param configuration
   * @param parameterObject 用户传入的实参
   */
  public DynamicContext(Configuration configuration, Object parameterObject) {
    // 参数不为空,为空后面会不会报错????
    if (parameterObject != null && !(parameterObject instanceof Map)) {
      MetaObject metaObject = configuration.newMetaObject(parameterObject);
      boolean existsTypeHandler = configuration
        .getTypeHandlerRegistry().hasTypeHandler(parameterObject.getClass());
      bindings = new ContextMap(metaObject, existsTypeHandler);
    } else {
      bindings = new ContextMap(null, false);
    }
    bindings.put(PARAMETER_OBJECT_KEY, parameterObject);
    bindings.put(DATABASE_ID_KEY, configuration.getDatabaseId());
  }

  public Map<String, Object> getBindings() {
    return bindings;
  }

  public void bind(String name, Object value) {
    bindings.put(name, value);
  }

  public void appendSql(String sql) {
    sqlBuilder.add(sql);
  }

  public String getSql() {
    return sqlBuilder.toString().trim();
  }

  public int getUniqueNumber() {
    return uniqueNumber++;
  }

  // 静态内部类,继承HashMap
  static class ContextMap extends HashMap<String, Object> {
    private static final long serialVersionUID = 2977601501966151582L;
    private final MetaObject parameterMetaObject;
    private final boolean fallbackParameterObject;

    /**
     * 公有构造方法,构造初始化自己的成员变量
     * @param parameterMetaObject
     * @param fallbackParameterObject
     */
    public ContextMap(MetaObject parameterMetaObject, boolean fallbackParameterObject) {
      this.parameterMetaObject = parameterMetaObject;
      this.fallbackParameterObject = fallbackParameterObject;
    }

    @Override
    public Object get(Object key) {
      String strKey = (String) key;
      // 判断数据库是否有了该数据
      if (super.containsKey(strKey)) {
        return super.get(strKey);
      }

      // 数据库没有数据,判断parameterMetaObject是否空
      if (parameterMetaObject == null) {
        return null;
      }
      
      // 上面parameterMetaObject非空,这里调用方法就不会出现NULL异常了
      if (fallbackParameterObject && !parameterMetaObject.hasGetter(strKey)) {
        return parameterMetaObject.getOriginalObject();
      } else {
        // issue #61 do not modify the context when reading
        return parameterMetaObject.getValue(strKey);
      }
    }
  }
}

如果对你有帮助,就点点赞,不要白嫖人家啦!!!讨厌!!

  大数据 最新文章
实现Kafka至少消费一次
亚马逊云科技:还在苦于ETL?Zero ETL的时代
初探MapReduce
【SpringBoot框架篇】32.基于注解+redis实现
Elasticsearch:如何减少 Elasticsearch 集
Go redis操作
Redis面试题
专题五 Redis高并发场景
基于GBase8s和Calcite的多数据源查询
Redis——底层数据结构原理
上一篇文章      下一篇文章      查看所有文章
加:2022-03-24 00:37:48  更:2022-03-24 00:40:25 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/16 15:36:22-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码