mybatis中实现让返回值与bean中字段相匹配
1. 编写目的
这个介绍的与那些修改mybatis.xml文件的方法不一样,目的也不一样。
当我们需要查询的数据跟entity的bean完全不匹配的时候(比如说需要统计的时候),我们不可能写多个dao层的查询接口,然后一个一个map到xml中去。
我们可以专门写一个类,根据自己的需要把统计的属性都写到里面去,然后一次性查询,就可以获得所有需要统计的数据。
2. 重要方法
专门编写一个实体类。实体类中包含的都是自己需要统计的属性。
编写dao层的接口方法的返回值就是这个实体类。
在映射的map.xml中编写查询时,使用as …的方法与实体类中的属性一一对应。
3. 具体案例
需求:需要统计日记表中某个用户的以下属性:
总共编写日记数目
删除日记数目
心情开心篇数
心情一般篇数
心情差篇数
心情极差篇数
晴朗天气篇数
阴天篇数
雨天篇数
…
实体类的编写
package cn.ailanglang.diary.util; public class StatisticBean { private Integer sum; private Integer mood0; private Integer mood1; private Integer mood2; private Integer mood3; private Integer weather0; private Integer weather1; private Integer weather2; private Integer weather3; private Integer weather4; private Integer weather5; private Integer weather6; private Integer weather7; private Integer unknow_weather; private Integer unknow_mood; public Integer getSum() { return sum; } public void setSum(Integer sum) { this.sum = sum; } public Integer getWeather0() { return weather0; } public void setWeather0(Integer weather0) { this.weather0 = weather0; } public Integer getWeather1() { return weather1; } public void setWeather1(Integer weather1) { this.weather1 = weather1; } public Integer getWeather2() { return weather2; } public void setWeather2(Integer weather2) { this.weather2 = weather2; } public Integer getWeather3() { return weather3; } public void setWeather3(Integer weather3) { this.weather3 = weather3; } public Integer getWeather4() { return weather4; } public void setWeather4(Integer weather4) { this.weather4 = weather4; } public Integer getWeather5() { return weather5; } public void setWeather5(Integer weather5) { this.weather5 = weather5; } public Integer getWeather6() { return weather6; } public void setWeather6(Integer weather6) { this.weather6 = weather6; } public Integer getWeather7() { return weather7; } public void setWeather7(Integer weather7) { this.weather7 = weather7; } public Integer getUnknow_weather() { return unknow_weather; } public void setUnknow_weather(Integer unknow_weather) { this.unknow_weather = unknow_weather; } public Integer getUnknow_mood() { return unknow_mood; } public void setUnknow_mood(Integer unknow_mood) { this.unknow_mood = unknow_mood; } public Integer getMood0() { return mood0; } public void setMood0(Integer mood0) { this.mood0 = mood0; } public Integer getMood1() { return mood1; } public void setMood1(Integer mood1) { this.mood1 = mood1; } public Integer getMood2() { return mood2; } public void setMood2(Integer mood2) { this.mood2 = mood2; } public Integer getMood3() { return mood3; } public void setMood3(Integer mood3) { this.mood3 = mood3; } }
dao层接口方法
/** * 统计 * @param userid * @return */ StatisticBean countMood(Long userid);
mapper.xml的编写
重点注意as …
<select id="countMood" parameterType="java.lang.Long" resultType="cn.smileyan.diary.util.StatisticBean"> select count(diary.pk_diaryid) as sum, sum(case when diary.mood='0' then 1 else 0 end) as mood0, sum(case when diary.mood='1' then 1 else 0 end) as mood1, sum(case when diary.mood='2' then 1 else 0 end) as mood2, sum(case when diary.mood='3' then 1 else 0 end) as mood3, sum(case when diary.weather='0' then 1 else 0 end) as weather0, sum(case when diary.weather='1' then 1 else 0 end) as weather1, sum(case when diary.weather='2' then 1 else 0 end) as weather2, sum(case when diary.weather='3' then 1 else 0 end) as weather3, sum(case when diary.weather='4' then 1 else 0 end) as weather4, sum(case when diary.weather='5' then 1 else 0 end) as weather5, sum(case when diary.weather='6' then 1 else 0 end) as weather6, sum(case when diary.weather='7' then 1 else 0 end) as weather7 from user_diary,diary where diary.pk_diaryid = user_diary.fk_diaryid and user_diary.fk_userid = #{userid}; </select>
编写service层就不再介绍了。
测试类方法
@Test public void test6() { StatisticBean statisticBean = diaryService.countMood((long) 25); System.out.println("sum=="+statisticBean.getSum()); System.out.println("mood(0) == "+statisticBean.getMood0()); System.out.println("mood(1) == "+statisticBean.getMood1()); System.out.println("mood(2) == "+statisticBean.getMood2()); System.out.println("mood(3) == "+statisticBean.getMood3()); System.out.println("weather(0) == "+statisticBean.getWeather0()); System.out.println("weather(1) == "+statisticBean.getWeather1()); System.out.println("weather(2) == "+statisticBean.getWeather2()); System.out.println("weather(3) == "+statisticBean.getWeather3()); System.out.println("weather(4) == "+statisticBean.getWeather4()); System.out.println("weather(5) == "+statisticBean.getWeather5()); System.out.println("weather(6) == "+statisticBean.getWeather6()); System.out.println("weather(7) == "+statisticBean.getWeather7()); }
可以成功输出数据库中的数据,完成了我们的目的――统计。
4. 总结
重点了解一下as 的用法,as后面跟着的是实体类的属性名,当然xml文件中一定要指明returnType是那个实体类,注意要写详细的class地址。
然后就是理解一下 sum(case when diary.mood=‘3' then 1 else 0 end) as mood3,中sum和case when的用法。
补充知识:mybatis 学习总结3 表字段与javabean字段的映射
有时候我们封装的javabean与库表的字段并不能一一对应,我们需要做一些必要的配置以保证数据能够正确的获取。
总的来说,解决库表与javabean字段不统一的方法有以下几种。
1.驼峰转换
我们在封装实体类的时候,通常将属性命令为驼峰形式,例如
userName
而库表的命名则遵循全小写,多个单词间使用 '_' 下划线连接的方式, 例如
user_name
这种情况我们可以使用mybatis的驼峰转换策略 在mybatis 的config.xml配置文件中 添加如下代码:
<!--设置集标签--> <settings> <!--设置标签 === 驼峰转换--> <setting name="mapUnderscoreToCamelCase" value="true"/> </settings>
添加如下策略后,针对查询结果集中的字段出现上例中的冲突时则能够得到解决。
2.sql语句中的 AS 关键字 ― 起别名
我们在数据库建一个course表
create table course ( id BIGINT KEY AUTO_INCREMENT, course_name VARCHAR(30) NOT NULL, grade SMALLINT NOT NULL, teacher_id BIGINT NOT NULL, add_time TIMESTAMP DEFAULT now(), mod_time TIMESTAMP DEFAULT now() )
插入几条数据
INSERT INTO course (course_name,grade,teacher_id) VALUES ('高等数学',1,1); INSERT INTO course (course_name,grade,teacher_id) VALUES ('微积分',2,2); INSERT INTO course (course_name,grade,teacher_id) VALUES ('希腊文学史',3,3);
建立实体类
@Alias("courseBO") public class CourseBO{ private Long courseId; //数据库-- id private String courseName; //数据库 -- course_name private Integer courseGrade; //数据库 -- grade private Long courseTeacherId; //数据库 -- teacher_id private Date addTime; //数据库 -- add_time private Date modTime; //数据库 -- mod_time /** * 以下为 get set 方法 以及 toString 方法 */ }
可以简单的发现几个库表与实体字段的不同
mapper.xml
<?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="mapping.CourseMapper"> <select id="getCourseByPriMaryKey" resultType="Model.CourseBO"> SELECT * FROM course WHERE id = #{id} </select> </mapper>
mapper接口
public interface CourseMapper extends BaseMapper { CourseBO getCourseByPriMaryKey(@Param("id") Long id); }
config.xml 注册mapper
<!--映射注册 加载映射文件--> <mappers> <!--第一种即第三种方法使用resource属性注册mapper,路径为xml文件--> <mapper resource="resource/mapper/UserMapper.xml" /> <!--第二种方式即注解方式使用class属性注册mapper,路径为接口class文件--> <!--<mapper class="mapping.UserMapper" />--> <mapper resource="resource/mapper/CourseMapper.xml" /> </mappers>
测试方法
public class FieldMappingDemo { public static void main(String[] args) throws IOException { String resource = "resource/common/mybatis-config.xml"; InputStream in = Resources.getResourceAsStream(resource); SqlSession sqlSession= new SqlSessionFactoryBuilder().build(in).openSession(); CourseMapper courseMapper = sqlSession.getMapper(CourseMapper.class); CourseBO courseBO = courseMapper.getCourseByPriMaryKey(1L); System.out.println(courseBO); sqlSession.close(); } }
执行结果
可以看到由于主键映射的失败,整个结果对象返回为null
我们现在在sql语句中加入字段别名,同时设置驼峰转换策略
<select id="getCourseByPriMaryKey" resultType="Model.CourseBO"> SELECT id courseId, course_name, grade courseGrade, teacher_id courseTeacherId, add_time, mod_time FROM course WHERE id = #{id} </select>
<!--设置集标签--> <settings> <!--设置标签 === 驼峰转换--> <setting name="mapUnderscoreToCamelCase" value="true"/> </settings>
再次执行,结果:
可以看到数据正确的获取到了。
3.resultMap建立字段映射
我们也可以使用resultMap做字段映射
<!--id:唯一标识 type:JavaBean实体--> <resultMap id="SqlMap" type="Model.CourseBO"> <!--id标签为主键标签,resultMap中必须存在一个id标签--> <id column="id" jdbcType="BIGINT" property="courseId" javaType="Long" /> <!--result标签为属性标签, column属性指定表字段名,jdbcType为表字段数据类型 property为实体属性名,javaType为实体属性数据类型--> <result column="course_name" jdbcType="VARCHAR" property="courseName" javaType="String"/> <result column="grade" jdbcType="SMALLINT" property="courseGrade" javaType="Integer"/> <result column="teacher_id" jdbcType="BIGINT" property="courseTeacherId" javaType="Long"/> <result column="add_time" jdbcType="TIMESTAMP" property="addTime" javaType="Date"/> <result column="mod_time" jdbcType="TIMESTAMP" property="modTime" javaType="Date"/> </resultMap> <!--select标签的 resultMap 指定resultMap标签中的id 值--> <!--去掉所有的别名 --> <select id="getCourseByPriMaryKey" resultMap="SqlMap"> SELECT * FROM course WHERE id = #{id} </select>
执行结果
阅读排行
- 1Java Swing组件BoxLayout布局用法示例
- 2java中-jar 与nohup的对比
- 3Java邮件发送程序(可以同时发给多个地址、可以带附件)
- 4Caused by: java.lang.ClassNotFoundException: org.objectweb.asm.Type异常
- 5Java中自定义异常详解及实例代码
- 6深入理解Java中的克隆
- 7java读取excel文件的两种方法
- 8解析SpringSecurity+JWT认证流程实现
- 9spring boot里增加表单验证hibernate-validator并在freemarker模板里显示错误信息(推荐)
- 10深入解析java虚拟机