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>

执行结果

以上这篇mybatis中实现让返回值与bean中字段相匹配就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持我们。

(0)

相关推荐

  • mybatis-generator自动生成dao、mapping、bean配置操作

    我就废话不多说了,大家还是直接看代码吧~ <?xml version="1.0" encoding="UTF-8"?> <!DOCTYPE generatorConfiguration PUBLIC "-//mybatis.org//DTD MyBatis Generator Configuration 1.0//EN" "http://mybatis.org/dtd/mybatis-generator-config_

  • 深入浅出重构Mybatis与Spring集成的SqlSessionFactoryBean(上)

    一般来说,修改框架的源代码是极其有风险的,除非万不得已,否则不要去修改.但是今天却小心翼翼的重构了Mybatis官方提供的与Spring集成的SqlSessionFactoryBean类,一来是抱着试错的心态,二来也的确是有现实需要. 先说明两点: 通常来讲,重构是指不改变功能的情况下优化代码,但本文所说的重构也包括了添加功能 本文使用的主要jar包(版本):spring-*-4.3.3.RELEASE.jar.mybatis-3.4.1.jar.mybatis-spring-1.3.0.jar

  • mybatis中实现让返回值与bean中字段相匹配

    1. 编写目的 这个介绍的与那些修改mybatis.xml文件的方法不一样,目的也不一样. 当我们需要查询的数据跟entity的bean完全不匹配的时候(比如说需要统计的时候),我们不可能写多个dao层的查询接口,然后一个一个map到xml中去. 我们可以专门写一个类,根据自己的需要把统计的属性都写到里面去,然后一次性查询,就可以获得所有需要统计的数据. 2. 重要方法 专门编写一个实体类.实体类中包含的都是自己需要统计的属性. 编写dao层的接口方法的返回值就是这个实体类. 在映射的map.x

  • MyBatis查询结果resultType返回值类型的说明

    一.返回一般数据类型 比如要根据 id 属性获得数据库中的某个字段值. mapper 接口: // 根据 id 获得数据库中的 username 字段的值 String getEmpNameById(Integer id); SQL 映射文件: <!-- 指定 resultType 返回值类型时 String 类型的, string 在这里是一个别名,代表的是 java.lang.String 对于引用数据类型,都是将大写字母转小写,比如 HashMap 对应的别名是 'hashmap' 基本数

  • JS在Chrome浏览器中showModalDialog函数返回值为undefined的解决方法

    本文实例讲述了JS在Chrome浏览器中showModalDialog函数返回值为undefined的解决方法.分享给大家供大家参考,具体如下: 主页面: <script type="text/javascript"> function SelectGroupCust() { var temp = window.showModalDialog("Default2.aspx?xx=" + Date(), "", "dialog

  • Android onTouchEvent事件中onTouch方法返回值(介绍)

    1.若return false说明没有成功执行onTouch事件,在执行完onTouch里面的代码之后,onTouch事件并没有结束.因此某些组件如Gallery会自动执行它所在view里onTouch方法的代码.若在onTouch方法里面增加你的代码并且最后return false就会执行你在OnTouch方法中的处理操作了. 2.若return true说明你已经成功执行onTouch方法了,在执行完onTouch中的代码之后,这个onTouch事件就结束了.也不会再调用组件如Gallery

  • Python中函数的返回值示例浅析

    前言: 前面我们介绍了简单的介绍了函数和函数的参数,今天我们来说一下Python中函数的返回值. 函数的返回值:函数运算的结果,需要进一步的操作时,给一个返回值return用来返回函数的结果,如果没有返回值,默认为None,python中可以间接返回多个值,也可以返回一个元组,程序在运行的时候,一旦遇到return,函数执行结束,后面的代码不会执行. def mypow(x,y=2): res = x**y print(res) return res print('python') mypow(

  • python中return不返回值的问题解析

    python中return不返回值是因为你没有将返回的值取出来. 解决方法: 调用函数,将函数的返回值赋给一个变量,输出这个变量就可以看到函数的返回值了 示例如下: def ss():     a = 10     b = 20     return a + b c = ss() print(c) 执行结果如下: 知识点扩展: return语句: return语句用来从一个函数 返回 即跳出函数.我们也可选从函数 返回一个值 . 使用字面意义上的语句 #!/usr/bin/python # Fi

  • Python中return函数返回值实例用法

    在学习return函数时候,还是要知道了解它最主要的函数作用,比如,怎么去实现返回一个值,另外还有就是我们经常会用到的使用return能够进行多值输出,这才是我们需要抓住知识的重点,针对上述所提及的内容,都可以来往下看文章,答案都在文章内容获取哦~ return 添加返回值 return 显示返回对象 返回值接受:value = func() 例子:计算学成最高分 listv = [90,80,88,77,66] # 分数计算return高分 def scoreCalculate(values)

  • java中关于return返回值的用法详解

    我们输入一个条件时,系统就会对这个条件进行判断,然后给出一个返回时的结论,我们把这个结果看做是返回值.在java里可以使用return语句来进行返回,从字面意思就能很好的理解它的用法了.下面我们就return的有无返回值进行分类展示,同时带来代码的实例分享. 1.定义 return语句可以使其从当前方法中退出,返回到调用该方法的语句处,继续程序的执行. 2.返回语句两种格式 有返回值: (1)return 返回值: (2)return 0 代表程序正常退出, (3)return 1 代表程序异常

  • 关于mybatis调用存储过程获取返回值问题

    总体思路:map传值 controller: Map<String,Object> m=new HashMap<String,Object>(); m.put("name", 'zs'); m.put("password", '55555'); cardservice.bindCard(m); JSONObject json=new JSONObject(); //获得返回值 json.put("msg", m.get(&

  • python中函数的返回值及类型详解

    目录 1.返回值介绍 2.带有返回值的函数 3.保存函数的返回值 4.四种函数的类型 1.无参数,无返回值的函数 2.无参数,有返回值的函数 3.有参数,无返回值的函数 4.有参数,有返回值的函数 5.小结 5.在python中我们可不可以返回多个值? 1.返回值介绍 现实生活中的场景: 我给儿子10块钱,让他给我买包烟.这个例子中,10块钱是我给儿子的,就相当于调用函数时传递到参数,让儿子买烟这个事情最终的目标是,让他把烟给你带回来然后给你对么,,,此时烟就是返回值 开发中的场景: 定义了一个

随机推荐