在Mybatis实现数据处理过程中,字段名符合数据库的规则,属性一般为驼峰规则,因此字段名和属性名通常不一致,此时可以通过以下两种方式对数据库字段进行映射处理:
- 为字段起别名,保证和实体类中的属性名一致
- 在满足驼峰转换规则时
在MyBatis的核心配置文件中设置一个全局配置信息mapUnderscoreToCamelCase,可以在查询表中数据时,自动将数据库字段名转换为驼峰 - 通过resultMap进行字段映射处理
1 利用字段别名映射
List<Emp> listEmp();
<!--通过字段别名解决字段名和属性名的映射问题--><select id="listEmp" resultType="Emp">select id, user_name as userName, pass_word as passWord, sex, dept_id as deptId from t_emp</select>
2 利用驼峰转换规则映射
在Mybatis核心配置文件中,设置全局的驼峰转换,此时框架会根据驼峰规则自动将数据库字段和实体属性进行映射。
- 核心配置
<settings><!-- 将表中的下划线字段自动映射为驼峰命名的属性字段 --><setting name="mapUnderscoreToCamelCase" value="true"/>
</settings>
- 数据查询
<!--通过驼峰配置,此时Mybatis框架会自动将字段和属性进行映射,如user_name映射到userName属性上。-->
<select id="listEmp" resultType="Emp">select id, user_name, pass_word, sex, dept_id from t_emp
</select>
此处resultType直接使用实体类别名,因为存在如下的配置:
<typeAliases><!--typeAlias:设置某个具体的类型的别名属性:type:需要设置别名的类型的全类名alias:设置此类型的别名若不设置此属性,该类型拥有默认的别名,即类名且不区分大小写; 若设置此属性,此时该类型的别名只能使用alias所设置的值--><!--<typeAlias type="com.giser.mybatis.bean.User"></typeAlias>--><!--<typeAlias type="com.giser.mybatis.bean.User" alias="abc"></typeAlias>--><!--以包为单位,设置改包下所有的类型都拥有默认的别名,即类名且不区分大小写--><package name="com.giser.pojo"/>
</typeAliases>
3 利用resultMap映射
<!--resultMap设置自定义映射关系id: 自定义映射的唯一标识type: 映射的实体类类型子标签:id: 设置主键的映射关系result:设置普通字段的映射关系属性:property: 设置映射关系中实体类的属性column: 设置映射关系中表字段--><resultMap id="empMap" type="Emp"><id property="id" column="id" /><result property="userName" column="user_name" /><result property="passWord" column="pass_word" /><result property="sex" column="sex" /><result property="deptId" column="dept_id" /></resultMap><!--使用ResultMap解决字段名和属性名的映射问题--><select id="listEmp" resultMap="empMap">select * from t_emp</select>
3.1 多对一映射
3.1.1 级联映射
存在如下实体:
public class Dept {private Integer id;private String deptName;private String deptNo;// getter setter
}public class EmpDept {private Integer id;private String userName;private String passWord;private String sex;private Integer deptId;private Dept dept;// getter setter
}
List<EmpDept> listEmpDeptById(@Param("eid")Integer eid);
<select id="listEmpDeptById" resultMap="empDeptMap">select * from t_emp left join t_dept on t_emp.dept_id = t_dept.id where t_emp.id = #{eid}
</select>
此时需要级联映射,如下:
<!--通过级联属性赋值
-->
<resultMap id="empDeptMap" type="EmpDept"><id property="id" column="id" /><result property="userName" column="user_name" /><result property="passWord" column="pass_word" /><result property="sex" column="sex" /><result property="deptId" column="dept_id" /><result property="dept.id" column="id" /><result property="dept.deptName" column="dept_name" /><result property="dept.deptNo" column="dept_no" />
</resultMap>
3.1.2 使用association实现多对一映射
如员工和部门的关系就是多对一,多个员工可能在同一个部门。此时可以通过上述的级联属性赋值方式进行映射,还可以通过association处理映射关系。
<!--通过association处理多对一映射关系
-->
<resultMap id="empDeptMapTwo" type="EmpDept"><id property="id" column="id" /><result property="userName" column="user_name" /><result property="passWord" column="pass_word" /><result property="sex" column="sex" /><result property="deptId" column="dept_id" /><!--association处理多对一的映射关系property: 需要处理多对一映射关系的实体属性名javaType: 实体属性的类型--><association property="dept" javaType="Dept"><id property="id" column="id" /><result property="deptName" column="dept_name" /><result property="deptNo" column="dept_no" /></association>
</resultMap><select id="listEmpDeptById" resultMap="empDeptMapTwo">select * from t_emp left join t_dept on t_emp.dept_id = t_dept.id where t_emp.id = #{eid}
</select>
3.1.3 使用association分步查询实现多对一映射
List<EmpDept> listEmpDeptByStep(@Param("eid") Integer eid);
<!--多对一映射关系处理方式二:通过分步查询
-->
<resultMap id="empDeptMapStep" type="EmpDept"><id property="id" column="id" /><result property="userName" column="user_name" /><result property="passWord" column="pass_word" /><result property="sex" column="sex" /><result property="deptId" column="dept_id" /><!--select :设置分步查询的Sql的唯一标识(namespace.SQLId或mapper接口的全类名.方法名)column :设置分步查询的条件--><association property="dept"
select="com.giser.mapper.DeptMapper.selectEmpAndDeptByStepTwo"column="dept_id"></association>
</resultMap><select id="listEmpDeptByStep" resultMap="empDeptMapStep">select * from t_emp left join t_dept on t_emp.dept_id = t_dept.id where t_emp.id = #{eid}
</select>
/*** 分步查询第二步* @param deptId* @return*/
Dept selectEmpAndDeptByStepTwo(@Param("deptId")Integer deptId);
<select id="selectEmpAndDeptByStepTwo" resultType="Dept">select * from t_dept where id = #{deptId}
</select>
- 延迟加载
分步查询的优点:可以实现延迟加载,但是必须在核心配置文件中设置全局配置信息:
lazyLoadingEnabled
:延迟加载的全局开关。当开启时,所有关联对象都会延迟加载
aggressiveLazyLoading
:当开启时,任何方法的调用都会加载该对象的所有属性。 否则,每个属性会按需加载
此时就可以实现按需加载,获取的数据是什么,就只会执行相应的sql。此时可通过association和collection中的fetchType
属性设置当前的分步查询是否使用延迟加载,fetchType=“lazy(延迟加载)|eager(立即加载)”。
在mybatis-config.xml中,开启懒加载配置
<settings><!-- 开启延迟加载 --><setting name="lazyLoadingEnabled" value="true" /></settings>
在映射关联属性时,设置fetchType=“lazy”
<!--多对一映射关系处理方式二:通过分步查询--><resultMap id="empDeptMapStep" type="EmpDept"><id property="id" column="id" /><result property="userName" column="user_name" /><result property="passWord" column="pass_word" /><result property="sex" column="sex" /><result property="deptId" column="dept_id" /><!--select :设置分布查询的Sql的唯一标识(namespace.SQLId或mapper接口的全类名.方法名)column :设置分布查询的条件fetchType: 延迟加载 ,只有当全局配置lazyLoadingEnabled设置为true时,fetchType=lazy才会延迟加载,否则都是立即加载fetchType="eager|lazy" eager:立即加载; lazy:延迟加载 且此处设置的优先级高于全局配置--><association property="dept"select="com.giser.mapper.DeptMapper.selectEmpAndDeptByStepTwo"column="dept_id"fetchType="lazy"></association></resultMap>
3.2 一对多映射
3.2.1 使用collection实现一对多映射
存在如下实体:
public class DeptEmp {private Integer id;private String deptName;private String deptNo;private List<Emp> emps;// getter setter
}
public class Emp {private Integer id;private String userName;private String passWord;private String sex;private Integer deptId;// getter setter
}
/*** 一对多关系查询* @param deptId* @return*/
DeptEmp selectDeptAndEmpByDeptId(@Param("deptId")Integer deptId);
<resultMap id="deptAndEmpResultMap" type="DeptEmp"><id property="id" column="id" /><result property="deptName" column="dept_name" /><result property="deptNo" column="dept_no" /><!--collection : 处理一对多关系映射property: 集合属性字段名称ofType : 表示该集合中存储的数据类型--><collection property="emps" ofType="Emp"><id property="id" column="id" /><result property="userName" column="user_name" /><result property="passWord" column="pass_word" /><result property="sex" column="sex" /></collection>
</resultMap><select id="selectDeptAndEmpByDeptId" resultMap="deptAndEmpResultMap">select * from t_dept left join t_emp on t_dept.id = t_emp.dept_id where t_dept.id = #{deptId}
</select>
3.2.2 使用分步查询实现一对多映射
DeptEmp selectDeptAndEmpByStepOne(@Param("deptId")Integer deptId);
<resultMap id="selectDeptAndEmpByStepResultMap" type="DeptEmp"><id property="id" column="id" /><result property="deptName" column="dept_name" /><result property="deptNo" column="dept_no" /><collection property="emps"
select="com.giser.mapper.EmpMapper.selectDeptAndEmpByStepTwo"column="id"></collection>
</resultMap><select id="selectDeptAndEmpByStepOne" resultMap="selectDeptAndEmpByStepResultMap">select * from t_dept where id = #{deptId}
</select>
Emp selectDeptAndEmpByStepTwo(@Param("deptId")Integer deptId);
<select id="selectDeptAndEmpByStepTwo" resultType="Emp">select * from t_emp where dept_id = #{deptId}
</select>