思考:如何解决属性名和字段名不一致时,数据为空的问题?
-
查看数据库中的字段
-
新建一个项目,测试实体类字段不一致的情况
public class User { private Integer id; private String name; private String password; // 数据库字段名为pwd //构造 //set/get //toString() }
-
编写接口文件
public interface UserMapper { //根据id查询用户 User getUserById(int id); }
-
编写mapper文件sql语句
<!-- 根据id查询用户 --> <select id="getUserById" parameterType="int" resultType="com.jh.domain.User"> select id,name,pwd from mybatis.user where id = #{id}; </select>
-
测试
public class UserMapperTest { @Test public void getUserById() { SqlSession sqlSession = MyBatisUtils.getSqlSession(); UserMapper userMapper = sqlSession.getMapper(UserMapper.class); User user = userMapper.getUserById(1); System.out.println(user); sqlSession.close(); } }
测试结果:
- User{id=1, name='小王', password='null'}
- 查询之后发现password为null,与预期不符
问题分析:
mybatis会根据这些查询的列名(会将列名转化为小写,数据库不区分大小写) , 去对应的实体类中查找相应列名的set方法设值 , 由于找不到setPwd() , 所以password返回null ; 【自动映射】
解决方案:在查询时为数据库字段名起别名
<!-- 根据id查询用户 -->
<select id="getUserById" parameterType="int" resultType="com.jh.domain.User">
select id,name,pwd as password
from mybatis.user
where id = #{id};
</select>
使用ResultMap解决上述问题
自动映射
resultMap
元素是 MyBatis 中最重要最强大的元素。它可以让你从 90% 的 JDBCResultSets
数据提取代码中解放出来,并在一些情形下允许你进行一些 JDBC 不支持的操作。- 在为一些比如连接的复杂语句编写映射代码的时候,一份
resultMap
能够代替实现同等功能的数千行代码。 - ResultMap 的设计思想是,对简单的语句做到零配置,对于复杂一点的语句,只需要描述语句之间的关系就行了。
简单映射语句的示例,它们没有显式指定 resultMap
。比如:
<select id="selectUsers" resultType="map">
select id, username, password
from mybatis
where id = #{id}
</select>
上述语句只是简单地将所有的列映射到 HashMap
的键上,这由 resultType
属性指定。虽然在大部分情况下都够用,但是 HashMap 不是一个很好的模型。你的程序更可能会使用 JavaBean 或 POJO(Plain Old Java Objects,普通老式 Java 对象)作为模型。
ResultMap
最优秀的地方在于,虽然你已经对它相当了解了,但是根本就不需要显式地用到他们。
手动映射
-
设置查询语句返回值为ResultMap
<!-- 根据id查询用户 --> <select id="getUserById" resultMap="userMap"> select * from mybatis.user where id = #{id}; </select>
-
编写ResultMap,实现手动映射
<!-- 结果集映射 --> <resultMap id="userMap" type="com.jh.domain.User"> <!-- id表示主键 --> <id property="id" column="id"/> <!-- column数据库中的字段,property实体类中的属性 --> <result property="name" column="name"/> <result property="password" column="pwd"/> </resultMap>
如果世界总是这么简单就好了。但是肯定不是的,数据库中,存在一对多,多对一的情况,我们之后会使用到一些高级的结果集映射,association,collection。
ResultMap的基本属性
id
:当前命名空间中的一个唯一标识,用于标识一个结果映射tyep
:类的完全限定名, 或者一个类型别名autoMapping
:如果设置这个属性,MyBatis 将会为本结果映射开启或者关闭自动映射。 这个属性会覆盖全局的属性 autoMappingBehavior。默认值:未设置(unset)。
ResultMap的子元素
constructor
- 用于在实例化类时,注入结果到构造方法中idArg
- ID 参数;标记出作为 ID 的结果可以帮助提高整体性能arg
- 将被注入到构造方法的一个普通结果
id
– 一个 ID 结果;标记出作为 ID 的结果可以帮助提高整体性能result
– 注入到字段或 JavaBean 属性的普通结果association
– 一个复杂类型的关联;许多结果将包装成这种类型- 嵌套结果映射 – 关联可以是
resultMap
元素,或是对其它结果映射的引用
- 嵌套结果映射 – 关联可以是
collection
– 一个复杂类型的集合- 嵌套结果映射 – 集合可以是
resultMap
元素,或是对其它结果映射的引用
- 嵌套结果映射 – 集合可以是
discriminator
– 使用结果值来决定使用哪个resultMap
case
– 基于某些值的结果映射- 嵌套结果映射 –
case
也是一个结果映射,因此具有相同的结构和元素;或者引用其它的结果映射
- 嵌套结果映射 –