mybatis某些字段无法映射成功的解决

 更新时间:2020年11月25日 11:02:51   作者:代码荷尔蒙  
这篇文章主要介绍了mybatis某些字段无法映射成功的解决方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧

随笔记录下:

刚刚遇到一个mybatis中reultMap定义正确column与property也都正确,字段的getset方法也都有,但是返回对象时,有些字段可以对应上有一些则不可以。

找了好久才发现在sql语句中的 resultMap 写成了 resultType。。。

很low但是痛。改成resultMap一切正常!

补充知识:MyBatis学习总结——解决字段名与实体类属性名不相同的冲突

在平时的开发中,我们表中的字段名和表对应实体类的属性名称不一定都是完全相同的,下面来演示一下这种情况下的如何解决字段名与实体类属性名不相同的冲突。

一、准备演示需要使用的表和数据

CREATE TABLE orders(
 order_id INT PRIMARY KEY AUTO_INCREMENT,
 order_no VARCHAR(20), 
 order_price FLOAT
);
INSERT INTO orders(order_no, order_price) VALUES('aaaa', 23);
INSERT INTO orders(order_no, order_price) VALUES('bbbb', 33);
INSERT INTO orders(order_no, order_price) VALUES('cccc', 22);

二、定义实体类

package me.gacl.domain;

/**
 * @author gacl
 * 定义orders表对应的实体类
 */
public class Order {
 /**
  * 
 CREATE TABLE orders(
  order_id INT PRIMARY KEY AUTO_INCREMENT,
  order_no VARCHAR(20), 
  order_price FLOAT
 );
 */
 
 //Order实体类中属性名和orders表中的字段名是不一样的
 private int id;    //id===>order_id
 private String orderNo;  //orderNo===>order_no
 private float price;  //price===>order_price

 public int getId() {
  return id;
 }

 public void setId(int id) {
  this.id = id;
 }

 public String getOrderNo() {
  return orderNo;
 }

 public void setOrderNo(String orderNo) {
  this.orderNo = orderNo;
 }

 public float getPrice() {
  return price;
 }

 public void setPrice(float price) {
  this.price = price;
 }

 @Override
 public String toString() {
  return "Order [id=" + id + ", orderNo=" + orderNo + ", price=" + price+ "]";
 }
}

三、编写测试代码

3.1、编写SQL的xml映射文件  

1、创建一个orderMapper.xml文件,orderMapper.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,namespace的值习惯上设置成包名+sql映射文件名,这样就能够保证namespace的值是唯一的
例如namespace="me.gacl.mapping.orderMapper"就是me.gacl.mapping(包名)+orderMapper(orderMapper.xml文件去除后缀)
 -->
<mapper namespace="me.gacl.mapping.orderMapper">
 
 <!-- 
  根据id查询得到一个order对象,使用这个查询是查询不到我们想要的结果的,
  这主要是因为实体类的属性名和数据库的字段名对应不上的原因,因此无法查询出对应的记录
 -->
 <select id="getOrderById" parameterType="int" 
  resultType="me.gacl.domain.Order">
  select * from orders where order_id=#{id}
 </select>
 
 <!-- 
  根据id查询得到一个order对象,使用这个查询是可以正常查询到我们想要的结果的,
  这是因为我们将查询的字段名都起一个和实体类属性名相同的别名,这样实体类的属性名和查询结果中的字段名就可以一一对应上
 -->
 <select id="selectOrder" parameterType="int" 
  resultType="me.gacl.domain.Order">
  select order_id id, order_no orderNo,order_price price from orders where order_id=#{id}
 </select>
 
 <!-- 
 根据id查询得到一个order对象,使用这个查询是可以正常查询到我们想要的结果的,
 这是因为我们通过<resultMap>映射实体类属性名和表的字段名一一对应关系 -->
 <select id="selectOrderResultMap" parameterType="int" resultMap="orderResultMap">
  select * from orders where order_id=#{id}
 </select>
 <!--通过<resultMap>映射实体类属性名和表的字段名对应关系 -->
 <resultMap type="me.gacl.domain.Order" id="orderResultMap">
  <!-- 用id属性来映射主键字段 -->
  <id property="id" column="order_id"/>
  <!-- 用result属性来映射非主键字段 -->
  <result property="orderNo" column="order_no"/>
  <result property="price" column="order_price"/>
 </resultMap>
 
</mapper>

2、在conf.xml文件中注册orderMapper.xml映射文件

<mappers>  
  <!-- 注册orderMapper.xml文件, 
  orderMapper.xml位于me.gacl.mapping这个包下,所以resource写成me/gacl/mapping/orderMapper.xml-->
  <mapper resource="me/gacl/mapping/orderMapper.xml"/>
</mappers>

3.2、编写单元测试代码

package me.gacl.test;

import me.gacl.domain.Order;
import me.gacl.util.MyBatisUtil;
import org.apache.ibatis.session.SqlSession;
import org.junit.Test;

public class Test2 {
 
 @Test
 public void testGetOrderById(){
  SqlSession sqlSession = MyBatisUtil.getSqlSession();
  /**
  * 映射sql的标识字符串,
  * me.gacl.mapping.orderMapper是orderMapper.xml文件中mapper标签的namespace属性的值,
  * getOrderById是select标签的id属性值,通过select标签的id属性值就可以找到要执行的SQL
  */
  String statement = "me.gacl.mapping.orderMapper.getOrderById";//映射sql的标识字符串
  //执行查询操作,将查询结果自动封装成Order对象返回
  Order order = sqlSession.selectOne(statement,1);//查询orders表中id为1的记录
  //使用SqlSession执行完SQL之后需要关闭SqlSession
  sqlSession.close();
  System.out.println(order);//打印结果:null,也就是没有查询出相应的记录
 }
 
 @Test
 public void testGetOrderById2(){
  SqlSession sqlSession = MyBatisUtil.getSqlSession();
  /**
  * 映射sql的标识字符串,
  * me.gacl.mapping.orderMapper是orderMapper.xml文件中mapper标签的namespace属性的值,
  * selectOrder是select标签的id属性值,通过select标签的id属性值就可以找到要执行的SQL
  */
  String statement = "me.gacl.mapping.orderMapper.selectOrder";//映射sql的标识字符串
  //执行查询操作,将查询结果自动封装成Order对象返回
  Order order = sqlSession.selectOne(statement,1);//查询orders表中id为1的记录
  //使用SqlSession执行完SQL之后需要关闭SqlSession
  sqlSession.close();
  System.out.println(order);//打印结果:Order [id=1, orderNo=aaaa, price=23.0]
 }
 
 @Test
 public void testGetOrderById3(){
  SqlSession sqlSession = MyBatisUtil.getSqlSession();
  /**
  * 映射sql的标识字符串,
  * me.gacl.mapping.orderMapper是orderMapper.xml文件中mapper标签的namespace属性的值,
  * selectOrderResultMap是select标签的id属性值,通过select标签的id属性值就可以找到要执行的SQL
  */
  String statement = "me.gacl.mapping.orderMapper.selectOrderResultMap";//映射sql的标识字符串
  //执行查询操作,将查询结果自动封装成Order对象返回
  Order order = sqlSession.selectOne(statement,1);//查询orders表中id为1的记录
  //使用SqlSession执行完SQL之后需要关闭SqlSession
  sqlSession.close();
  System.out.println(order);//打印结果:Order [id=1, orderNo=aaaa, price=23.0]
 }
}

执行单元测试的结果:

1、testGetOrderById方法执行查询后返回一个null。

2、testGetOrderById2方法和testGetOrderById3方法执行查询后可以正常得到想要的结果。

四、总结   

上面的测试代码演示当实体类中的属性名和表中的字段名不一致时,使用MyBatis进行查询操作时无法查询出相应的结果的问题以及针对问题采用的两种办法:

解决办法一:

通过在查询的sql语句中定义字段名的别名,让字段名的别名和实体类的属性名一致,这样就可以表的字段名和实体类的属性名一一对应上了,这种方式是通过在sql语句中定义别名来解决字段名和属性名的映射关系的。

解决办法二:

通过<resultMap>来映射字段名和实体类属性名的一一对应关系。这种方式是使用MyBatis提供的解决方式来解决字段名和属性名的映射关系的。

以上这篇mybatis某些字段无法映射成功的解决就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • 详解Java 连接MongoDB集群的几种方式

    详解Java 连接MongoDB集群的几种方式

    这篇文章主要介绍了详解Java 连接MongoDB集群的几种方式,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-12-12
  • 通过端口1433连接到主机127.0.0.1的 TCP/IP 连接失败,错误:“connect timed out”的解决方法

    通过端口1433连接到主机127.0.0.1的 TCP/IP 连接失败,错误:“connect timed out”的解

    这篇文章主要介绍了通过端口1433连接到主机127.0.0.1的 TCP/IP 连接失败,错误:“connect timed out”的解决方法,需要的朋友可以参考下
    2015-08-08
  • java中1+1d/5和1+1/5的区别说明

    java中1+1d/5和1+1/5的区别说明

    这篇文章主要介绍了java中1+1d/5和1+1/5的区别说明,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-10-10
  • java 多线程死锁详解及简单实例

    java 多线程死锁详解及简单实例

    这篇文章主要介绍了java 多线程死锁详解及简单实例的相关资料,需要的朋友可以参考下
    2017-01-01
  • JavaMailSender实现邮箱验证功能

    JavaMailSender实现邮箱验证功能

    本篇文章主要给大家介绍了JavaMailSender实现邮箱注册验证的功能实现原理以及其中遇到的问题,一起跟着学习探讨下吧。
    2017-12-12
  • Java如何利用策略模式替代if/else语句

    Java如何利用策略模式替代if/else语句

    这篇文章主要介绍了Java如何利用策略模式替代if/else语句,帮助大家优化自己的代码,提高程序运行效率,感兴趣的朋友可以了解下
    2020-09-09
  • Java数据结构之LinkedList从链表到实现

    Java数据结构之LinkedList从链表到实现

    LinkedList是Java中常用的数据结构之一,实现了链表的特性,支持快速添加、删除元素,可以用于实现队列、栈、双向队列等数据结构。LinkedList的内部实现采用了双向链表,其中每个节点都包含前驱节点和后继节点的引用,可以直接访问链表的头尾元素
    2023-04-04
  • Spring mvc工作原理_动力节点Java学院整理

    Spring mvc工作原理_动力节点Java学院整理

    这篇文章主要为大家详细介绍了Spring mvc工作原理的相关资料,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-08-08
  • 简单了解java标识符的作用和命名规则

    简单了解java标识符的作用和命名规则

    这篇文章主要介绍了简单了解java标识符的作用和命名规则,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-01-01
  • java 跳转搜索的实现示例

    java 跳转搜索的实现示例

    与二分搜索一样,跳转搜索是一种针对排序数组的搜索算法,本文主要介绍了java 跳转搜索的实现示例,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起学习学习吧
    2024-04-04

最新评论