6千字带你看懂Mybatis字段映射

您所在的位置:网站首页 mybatis映射关系的问题讨论 6千字带你看懂Mybatis字段映射

6千字带你看懂Mybatis字段映射

#6千字带你看懂Mybatis字段映射| 来源: 网络整理| 查看: 265

前言

考虑到在Select时使用AS和方案一其实没什么差别,在介绍ResultMap之前,顺便带过一下。

方案二-Select … AS

当我们的数据库列名和对象字段之间不是驼峰式命名的关系,我们可以在Select时使用AS,使得列名和对象名匹配上。 映射文件中是本次会执行的sql,我们会查出id,city_id,city_name,city_en_name。 按照开启的驼峰式命名开关,我们会对应到对象的id,cityId,cityName,cityEnName字段。

select id,city_id,city_name,city_en_name from SU_City where id = #{id}

不过在这次,我们对PO做了小小的改动,把cityEnName改成了cityEnglishName。

public class CityPO { Integer id; Long cityId; String cityName; String cityEnglishName; // 由cityEnName改成了cityEnglishName

由于找不到匹配的列,cityEnlishName肯定没法被反射赋值,要为Null了。

CityPO{id=2, cityId=2, cityName='北京', cityEnglishName='null'}

**解决办法: **在Select字段的时候使用AS,下面是改动后的映射文件。

select id, city_id, city_name, city_en_name AS cityEnglishName from SU_City where id = #{id}

改动后执行得到的结果如下。

CityPO{id=2, cityId=2, cityName='北京', cityEnglishName='beijing'}

那么我们来看看它是如何生效的,主要的代码在哪里。在昨天我们第一个介绍的函数handleRowValues中传入了参数rsw,它是对ResultSet的一个包装,在这个包装里,完成了具体使用哪个名字作为数据库的列名。

final ResultSetWrapper rsw = new ResultSetWrapper(rs, configuration); handleRowValues(rsw, resultMap, resultHandler, new RowBounds(), null);

在这个构造函数当中,我们会获取数据库的列名,AS为什么可以生效,具体就在下面这段代码。

super(); this.typeHandlerRegistry = configuration.getTypeHandlerRegistry(); this.resultSet = rs; final ResultSetMetaData metaData = rs.getMetaData(); final int columnCount = metaData.getColumnCount(); for (int i = 1; i

在这个函数中,会获取没有映射过的列名。

final List unmappedColumnNames = rsw.getUnmappedColumnNames(resultMap, columnPrefix);

之后会根据resultMap查看是否有未映射的字段。

loadMappedAndUnmappedColumnNames(resultMap, columnPrefix); List mappedColumnNames = new ArrayList(); List unmappedColumnNames = new ArrayList(); final String upperColumnPrefix = columnPrefix == null ? null : columnPrefix.toUpperCase(Locale.ENGLISH); // 这里没有配置前缀,根据之前的图,定义了ResultMap后,会记录这些已经配置映射的字段。 final Set mappedColumns = prependPrefixes(resultMap.getMappedColumns(), upperColumnPrefix); for (String columnName : columnNames) { // 遍历列名,如果在已映射的配置中,那么就加入已经映射的列名数据, final String upperColumnName = columnName.toUpperCase(Locale.ENGLISH); if (mappedColumns.contains(upperColumnName)) { mappedColumnNames.add(upperColumnName); } else { unmappedColumnNames.add(columnName); } } // 生成未映射和已映射的Map mappedColumnNamesMap.put(getMapKey(resultMap, columnPrefix), mappedColumnNames); unMappedColumnNamesMap.put(getMapKey(resultMap, columnPrefix), unmappedColumnNames);

如果有没配置在ResultMap中,且Select出来的,那么之后也会按照之前方案一那样,继续往下走,去对象中寻找映射关系。 由于没有未映射的字段,使用自动映射的结果是false。

foundValues = applyAutomaticMappings(rsw, resultMap, metaObject, null) || foundValues;

之后继续往下走,使用applyPropertyMappings来创建对象。使用了PropertyMapping。里面包含了字段名,列名,字段的类型和对应的处理器。

遍历整个Mappings。

Object value = getPropertyMappingValue(rsw.getResultSet(), metaObject, propertyMapping, lazyLoader, columnPrefix);

函数里主要的就是获取这个字段对应的类型处理器,防止类型转换失败,这一部分下次会专门看一下。

final TypeHandler typeHandler = propertyMapping.getTypeHandler(); final String column = prependPrefix(propertyMapping.getColumn(), columnPrefix); return typeHandler.getResult(rs, column);

TypeHandler就是一个接口,主要完成的工作就是从Result根据列名,获取相应类型的值,为下一步反射赋值做准备。至于它是怎么决定为什么用这个类型的TypeHandler下次再看。最后就是给对应字段赋值。

metaObject.setValue(property, value);

最后就完成了整个类的赋值。

总结

大致上,Mybatis完成映射主要是两种方式。

只根据列名,利用自动映射,根据反射类的信息,得到列名和字段之间的关系,使用对应的TypeHandler,完成字段的赋值。

使用ResultMap预先定义好映射关系,也是最后根据TypeHandler和反射,完成字段的赋值。 我个人感觉就简单的用法来说,两者都可以,在一次会话中,Configuration中的ResultMap关系建立好,在每一次查询的时候就不用再去重新建立了,直接用就行。而自动映射的话,执行过一次后,也会在会话中建立自动映射的缓存。所以没什么差别。但如果复杂的映射的话,就非ResultMap莫属啦。具体可以参考Mybatis文档关于映射的章节,因为目前用不到比较复杂的映射, 不做深究了。

http://www.mybatis.org/mybatis-3/zh/sqlmap-xml.html



【本文地址】


今日新闻


推荐新闻


CopyRight 2018-2019 办公设备维修网 版权所有 豫ICP备15022753号-3