在上篇博客-[[JDBC] 处理ResultSet,构建Java对象](https://my.oschina.net/kailun...中提到,咱们须要分析Mybatis在转换Result到须要的Java业务对象时作的三件事,以下:java
其实核心就是:mysql
今天咱们先来看第一点,数据库中的列名怎么和对象中的字段对应起来。首先是平常PO(Persistant Object) CityPO,里面有五个字段。sql
public class CityPO { Integer id; Long cityId; String cityName; String cityEnName; String cityPyName;
本次要查询的数据库中的列名以下所示。数据库
mysql> mysql> desc SU_City; +--------------+-------------+------+-----+-------------------+-----------------------------+ | Field | Type | Null | Key | Default | Extra | +--------------+-------------+------+-----+-------------------+-----------------------------+ | id | int(11) | NO | PRI | NULL | auto_increment | | city_id | int(11) | NO | UNI | NULL | | | city_name | varchar(20) | NO | | | | | city_en_name | varchar(20) | NO | | | | | city_py_name | varchar(50) | NO | | | | | create_time | datetime | NO | | CURRENT_TIMESTAMP | | | updatetime | datetime | NO | MUL | CURRENT_TIMESTAMP | on update CURRENT_TIMESTAMP | +--------------+-------------+------+-----+-------------------+-----------------------------+ 7 rows in set (0.01 sec)
咱们是按照驼峰式命名,把数据库中的列名对应到了对象的字段名。以下是Mybatis的接口类和映射文件。bash
public interface CityMapper { CityPO selectCity(int id); }
<?xml version="1.0" encoding="UTF-8" ?> <!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" "http://mybatis.org/dtd/mybatis-3-mapper.dtd"> <mapper namespace="mapper.CityMapper"> <select id="selectCity" resultType="po.CityPO"> select id,city_id,city_name,city_en_name from SU_City where id = #{id} </select> </mapper>
在上面的映射文件中,namespace指定了这个接口类的全限定类名,紧随其后的select表明是select语句,id是接口类中函数的名字,resultType表明了从这条语句中返回的指望类型的类的彻底限定名或别名,在此例子中是咱们的业务对象CityPO的类路径。微信
主要有三种方案mybatis
这篇主要看一下第一种,附上示例和部分源码走读。app
<settings> <!-- 开启驼峰,开启后,只要数据库字段和对象属性名字母相同,不管中间加多少下划线均可以识别 --> <setting name="mapUnderscoreToCamelCase" value="true" /> </settings>
咱们从源码角度解读一下,Mybat处理ResultSet的映射默认都在DefaultResultSetHandler中完成。
处理行数据的时候的时候主要在下面?的函数里进行,因为咱们在映射文件中没有定义额外的ResultMap,所以会直接进入else分支的代码。函数
public void handleRowValues(ResultSetWrapper rsw, ResultMap resultMap, ResultHandler<?> resultHandler, RowBounds rowBounds, ResultMapping parentMapping) throws SQLException { if (resultMap.hasNestedResultMaps()) { ensureNoRowBounds(); checkResultHandler(); handleRowValuesForNestedResultMap(rsw, resultMap, resultHandler, rowBounds, parentMapping); } else { handleRowValuesForSimpleResultMap(rsw, resultMap, resultHandler, rowBounds, parentMapping); } }
进入handleRowValuesForSimpleResultMap中,主要处理函数以下,在这里完成了对象的生成及赋值。ui
Object rowValue = getRowValue(rsw, discriminatedResultMap);
在这里先建立了对象的实例,而后获取了对象的元信息,为反射赋值作准备。
private Object getRowValue(ResultSetWrapper rsw, ResultMap resultMap) throws SQLException { final ResultLoaderMap lazyLoader = new ResultLoaderMap(); Object rowValue = createResultObject(rsw, resultMap, lazyLoader, null); if (rowValue != null && !hasTypeHandlerForResultObject(rsw, resultMap.getType())) { final MetaObject metaObject = configuration.newMetaObject(rowValue); boolean foundValues = this.useConstructorMappings; if (shouldApplyAutomaticMappings(resultMap, false)) { foundValues = applyAutomaticMappings(rsw, resultMap, metaObject, null) || foundValues; } foundValues = applyPropertyMappings(rsw, resultMap, metaObject, lazyLoader, null) || foundValues; foundValues = lazyLoader.size() > 0 || foundValues; rowValue = (foundValues || configuration.isReturnInstanceForEmptyRow()) ? rowValue : null; } return rowValue; }
在applyAutomaticMappings完成了整个过程,咱们进去探一探。
就是下面这个函数建立好了映射关系,这个函数的下半部分是完成赋值的,映射的部分下次会详细分析。
List<UnMappedColumnAutoMapping> autoMapping = createAutomaticMappings(rsw, resultMap, metaObject, columnPrefix);
在这个方法里,上半部分是生成了数据库的列名,在这个函数中找到了对应的字段名。
final String property = metaObject.findProperty(propertyName, configuration.isMapUnderscoreToCamelCase());
咱们进去看一看,它传进了生成好的数据库列名,传进了前面提到的是否根据驼峰式命名映射开关的值。
事实证实,真的很简单,往下看,就是把下划线都去了。
public String findProperty(String name, boolean useCamelCaseMapping) { if (useCamelCaseMapping) { name = name.replace("_", ""); } return findProperty(name); }
隐隐以为是否是大小写不敏感啊,继续往下看,这里返回找到的字段名。
private StringBuilder buildProperty(String name, StringBuilder builder) { .......... String propertyName = reflector.findPropertyName(name); if (propertyName != null) { builder.append(propertyName); } } return builder; }
好了,真相大白,就是大小写不敏感的。
public String findPropertyName(String name) { return caseInsensitivePropertyMap.get(name.toUpperCase(Locale.ENGLISH)); }
因此若是你数据库里字段是city_id,city_Id,大写I,那么可能会有问题吧,不过仔细想一想,谁会吃力不讨好干这种事情,硬要处理成标准的驼峰式命名也能够啦,不过感受必要性不大。
通过若干次中途崩溃,我终于写完了驼峰式命名开关下,咱们是如何完成数据库列和字段名的映射的。后面的博文会继续看看后续两种方案以及DDL时对象字段是如何赋值到Sql语句中。
若是想进一步了解的话,欢迎关注个人微信公众号