Mybatis四种实例化对象方式

代码准备

创建mybatis-config.xml
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE configurationPUBLIC "-//mybatis.org//DTD Config 3.0//EN""http://mybatis.org/dtd/mybatis-3-config.dtd">
<configuration><properties><property name="driver" value="com.mysql.cj.jdbc.Driver"/><property name="url" value="jdbc:mysql://127.0.0.1:3306/test?useUnicode=true&amp;characterEncoding=utf-8&amp;useSSL=false&amp;serverTimezone=Asia/Shanghai&amp;allowPublicKeyRetrieval=true"/><property name="username" value="root"/><property name="password" value="123456"/></properties><settings><setting name="mapUnderscoreToCamelCase" value="true"/></settings><environments default="default"><environment id="default"><transactionManager type="JDBC"/><dataSource type="POOLED"><property name="driver" value="${driver}"/><property name="url" value="${url}"/><property name="username" value="${username}"/><property name="password" value="${password}"/></dataSource></environment></environments><mappers><mapper resource="mapper/ConstructorMapper.xml" /></mappers></configuration>
创建ConstructorMapper.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="com.ys.mybatis.mapper.ConstructorMapper"><resultMap id="employeeMap" type="com.ys.mybatis.domian.Employee"><constructor><idArg name="id" column="id"/><arg name="name" column="name" /></constructor><result property="age" column="age"/><result property="phone" column="phone"/></resultMap><select id="getEmployeeByIdForSimple" resultType="com.ys.mybatis.domian.Employee">select * from employee where id = #{id}</select><select id="getEmployeeByIdForResultMap" resultMap="employeeMap">select * from employee where id = #{id}</select></mapper>
创建实体类Employee 
@Slf4j
@Data
public class Employee {private Integer id;private String name;private Integer age;private String phone;public Employee() {log.info("调用了空构造方法");}public Employee(Integer id) {this.id = id;log.info("调用了一个参数的构造方法");}public Employee(Integer id, String name) {this.id = id;this.name = name;log.info("调用了两个参数的构造方法");}public Employee(Integer id, String name, Integer age) {this.id = id;this.name = name;this.age = age;log.info("调用了三个参数的构造方法");}public Employee(Integer id, String name, Integer age, String phone) {this.id = id;this.name = name;this.age = age;this.phone = phone;log.info("调用了四个参数的构造方法");}
}
创建ConstructorMapper.java
public interface ConstructorMapper {Employee getEmployeeByIdForSimple(Integer id);Employee getEmployeeByIdForResultMap(Integer id);
}
创建测试类ConstructorTest 
@Slf4j
public class ConstructorTest {private SqlSessionFactory sqlSessionFactory;@BeforeEachpublic void before() {InputStream inputStream = ConfigurationTest.class.getClassLoader().getResourceAsStream("mybatis-config.xml");sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);}
}

1.使用默认构造方法

@Test
public void testDefaultConstructor() {// 获取sqlSessionSqlSession sqlSession = sqlSessionFactory.openSession();// 获取ConstructorMapper对象ConstructorMapper mapper = sqlSession.getMapper(ConstructorMapper.class);Employee employee = mapper.getEmployeeByIdForSimple(1);System.out.println(employee);
}

默认情况下,mybatis通过默认构造方法,实例化对象

2.使用指定构造方法

@Test
public void testResultMap() {// 获取sqlSessionSqlSession sqlSession = sqlSessionFactory.openSession();// 获取ConstructorMapper对象ConstructorMapper mapper = sqlSession.getMapper(ConstructorMapper.class);Employee employee = mapper.getEmployeeByIdForResultMap(1);System.out.println(employee);
}

如果select指定了resultMap,且resultMap存在constructor标签,则mybatis就根据constructor标签指定的构造器进行实例化

 PS : 如果测试方法报错,需要在maven文件中添加如下配置:

<plugins><plugin><groupId>org.apache.maven.plugins</groupId><artifactId>maven-compiler-plugin</artifactId><version>3.6.2</version><configuration><parameters>true</parameters></configuration></plugin>
</plugins>

3.没有默认构造方法,ResultMap也不存在constructor标签 (将默认构造方法注释)

相关sql、Mapper接口方法、测试方法、测试结果 如下所示:

<resultMap id="noDefaultEmployeeMap" type="com.ys.mybatis.domian.Employee"><result property="id" column="id"/><result property="name" column="name"/><result property="age" column="age"/><result property="phone" column="phone"/>
</resultMap><select id="getEmployeeByIdForNoDefaultResultMap" resultMap="noDefaultEmployeeMap">select id,name,age,phone from employee where id = #{id}
</select>
Employee getEmployeeByIdForNoDefaultResultMap(Integer id);
@Test
public void noDefaultConstructorTest() {// 获取sqlSessionSqlSession sqlSession = sqlSessionFactory.openSession();// 获取ConstructorMapper对象ConstructorMapper mapper = sqlSession.getMapper(ConstructorMapper.class);Employee employee = mapper.getEmployeeByIdForNoDefaultResultMap(1);System.out.println(employee);
}

选择参数个数和返回列数相等的构造方法

PS : 注意是查询返回的列数,不是ResultMap指定的映射个数。如果将查询的列数改成三个,则使用三个参数的构造方法

<select id="getEmployeeByIdForNoDefaultResultMap" resultMap="noDefaultEmployeeMap">select id,name,age from employee where id = #{id}
</select>

4.自定义TypeHandler

4.1 自定义EmployeeTypeHandler
@MappedTypes(value = {Employee.class})
public class EmployeeTypeHandler extends BaseTypeHandler<Employee> {@Overridepublic void setNonNullParameter(PreparedStatement ps, int i, Employee employee, JdbcType jdbcType) throws SQLException {switch (i) {case 1:ps.setInt(i, employee.getId());break;case 2:ps.setString(i, employee.getName());break;case 3:ps.setInt(i, employee.getAge());break;case 4:ps.setString(i, employee.getPhone());break;}}@Overridepublic Employee getNullableResult(ResultSet rs, String columnName) throws SQLException {Employee employee = new Employee();employee.setId(rs.getInt("id"));employee.setName(rs.getString("name"));employee.setAge(rs.getInt("age"));employee.setPhone(rs.getString("phone"));return employee;}@Overridepublic Employee getNullableResult(ResultSet rs, int columnIndex) throws SQLException {Employee employee = new Employee();employee.setId(rs.getInt(1));employee.setName(rs.getString(2));employee.setAge(rs.getInt(3));employee.setPhone(rs.getString(4));return employee;}@Overridepublic Employee getNullableResult(CallableStatement cs, int columnIndex) throws SQLException {Employee employee = new Employee();employee.setId(cs.getInt(1));employee.setName(cs.getString(2));employee.setAge(cs.getInt(3));employee.setPhone(cs.getString(4));return employee;}
}
4.2 在mybatis-config.xml文件中配置TypeHandler
<typeHandlers><typeHandler handler="com.ys.mybatis.handler.EmployeeTypeHandler"/>
</typeHandlers>
4.3 执行测试方法
@Test
public void testDefaultConstructor() {// 获取sqlSessionSqlSession sqlSession = sqlSessionFactory.openSession();// 获取ConstructorMapper对象ConstructorMapper mapper = sqlSession.getMapper(ConstructorMapper.class);Employee employee = mapper.getEmployeeByIdForSimple(1);System.out.println(employee);
}

自定义TypeHandler可以使用任意构造方法实例化对象,比如将getNullableResult方法修改为如下所示:

@Override
public Object getNullableResult(ResultSet rs, String columnName) throws SQLException {return new Employee(rs.getInt("id"), rs.getString("name"), rs.getInt("age"), rs.getString("phone"));
}

局部使用typeHandler

假设现在Employee存在一个additional属性,其在数据库中是以Json格式存储的,我们希望可以将这个Json字符串,转成Additional实例对象。Additional结构如下:

@NoArgsConstructor
@AllArgsConstructor
@Data
public class Additional {private String email;private String address;}
创建AdditionalTypeHandler
public class AdditionalTypeHandler extends BaseTypeHandler<Additional> {@Overridepublic void setNonNullParameter(PreparedStatement ps, int i, Additional additional, JdbcType jdbcType) throws SQLException {switch (i) {case 1:ps.setString(i, additional.getEmail());break;case 2:ps.setString(i, additional.getAddress());break;}}@Overridepublic Additional getNullableResult(ResultSet rs, String columnName) throws SQLException {String val = rs.getString(columnName);return new Gson().fromJson(val, Additional.class);}@Overridepublic Additional getNullableResult(ResultSet rs, int columnIndex) throws SQLException {String val = rs.getString(columnIndex);return new Gson().fromJson(val, Additional.class);}@Overridepublic Additional getNullableResult(CallableStatement cs, int columnIndex) throws SQLException {String val = cs.getString(columnIndex);return new Gson().fromJson(val, Additional.class);}
}
相关sql、Mapper接口方法、测试方法、测试结果 如下所示:
<resultMap id="additionalMap" type="com.ys.mybatis.domian.Employee"><result property="id" column="id"/><result property="name" column="name"/><result property="age" column="age"/><result property="phone" column="phone"/><result property="additional" column="json" typeHandler="com.ys.mybatis.handler.AdditionalTypeHandler"/>
</resultMap><select id="getEmployeeByIdForPart" resultMap="additionalMap">select * from employee where id = #{id} 
</select>
Employee getEmployeeByIdForPart(Integer id);
@Test
public void testPartTypeHandler() {// 获取sqlSessionSqlSession sqlSession = sqlSessionFactory.openSession();// 获取ConstructorMapper对象ConstructorMapper mapper = sqlSession.getMapper(ConstructorMapper.class);Employee employee = mapper.getEmployeeByIdForPart(1);System.out.println(employee);
}

PS : 需要把步骤4.2在mybatis-config.xml文件中配置的EmployeeTypeHandler移除

优先级

  1. 返回类型存在自定义的TypeHandler
  2. ResultMap中指定了构造方法
  3. 默认构造方法
  4. 无默认构造方法,且ResultMap中未指定了构造方法 (resultType也会被解析成ResultMap)

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/661835.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

深度学习之基于YOLOv5智慧交通拥挤预警检测系统

欢迎大家点赞、收藏、关注、评论啦 &#xff0c;由于篇幅有限&#xff0c;只展示了部分核心代码。 文章目录 一项目简介 二、功能三、系统四. 总结 一项目简介 一、项目背景 随着城市化进程的加速和人口规模的不断增长&#xff0c;交通拥挤问题日益严重。传统的交通拥挤预警方…

阿里云开源大模型开发环境搭建

ModelScope是阿里云通义千问开源的大模型开发者社区&#xff0c;本文主要描述AI大模型开发环境的搭建。 如上所示&#xff0c;安装ModelScope大模型基础库开发框架的命令行参数&#xff0c;使用清华大学提供的镜像地址 如上所示&#xff0c;在JetBrains PyCharm的项目工程终端控…

POETIZE个人博客系统源码 | 最美博客

源码介绍 POETIZE个人博客系统源码 | 最美博客 这是一个 SpringBoot Vue2 Vue3 的产物&#xff0c;支持移动端自适应&#xff0c;配有完备的前台和后台管理功能。 网站分两个模块&#xff1a; 博客系统&#xff1a;具有文章&#xff0c;表白墙&#xff0c;图片墙&#xf…

FFmpeg开发笔记(二十三)使用OBS Studio开启RTMP直播推流

OBS是一个开源的直播录制软件&#xff0c;英文全称叫做Open Broadcaster Software&#xff0c;广泛用于视频录制、实时直播等领域。OBS不但开源&#xff0c;而且跨平台&#xff0c;兼容Windows、Mac OS、Linux等操作系统。 OBS的官网是https://obsproject.com/&#xff0c;录制…

数据结构—C语言实现双向链表

目录 1.双向带头循环链表 2.自定义头文件&#xff1a; 3.List.cpp 文件 3.1 newnode()函数讲解 3.2 init() 函数 初始化 3.3 pushback()函数 尾插 3.4 pushfront()函数 头插 3.5 popback() 尾删 3.6 popfront() 函数 头删 3.7 insert()函数 在pos之后插入 3.8 popbac…

git 配置相关

问题一&#xff1a;ssh-keygen -t ed25519 -C "Gitee SSH Key" 这个命令中的 ed25519 字符是什么意思&#xff1f; ssh-keygen 是一个用于生成SSH密钥的工具&#xff0c;SSH&#xff08;Secure Shell&#xff09;是一种网络协议&#xff0c;用于加密方式远程登录和其…

easyExcel - 带图片导出

目录 前言一、情景介绍二、问题分析三、代码实现1. 单图片导出2. 多图片导出3. 多图片导出&#xff08;优化&#xff09; 前言 Java-easyExcel入门教程&#xff1a;https://blog.csdn.net/xhmico/article/details/134714025 之前有介绍过如何使用 easyExcel&#xff0c;以及写…

24.什么是跨域?解决方案有哪些?

为什么会出现跨域问题 存在浏览器同源策略&#xff0c;所以才会有跨域问题。那么浏览器是出于何种原因会有跨域的限制呢。其实不难想到&#xff0c;跨域限制主要的目的就是为了用户的上网安全。 同源策略导致的跨域是浏览器单方面拒绝响应数据&#xff0c;服务器端是处理完毕…

java+jsp+Oracle+Tomcat 记账管理系统论文(二)

⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️⬇️ ➡️点击免费下载全套资料:源码、数据库、部署教程、论文、答辩ppt一条龙服务 ➡️有部署问题可私信联系 ⬆️⬆️⬆️​​​​​​​⬆️…

从0到1手写注册中心Registry之主从数据同步

集群选主完成后&#xff0c;从节点需要从主节点同步数据。 一、快照数据 同步的快照数据是描述注册中心的服务信息。 registry: 每个服务对应的实例&#xff1b;versions&#xff1a;每个服务的版本信息&#xff1b;timestamps&#xff1a;每个服务的时间戳&#xff1b;vers…

工作任务管理平台B端实战项目作品集+WebApp项目源文件 figma格式

首先&#xff0c;作品集是什么&#xff1f;通常应该包含什么内容&#xff1f;为什么大家都在做自己的作品集呢&#xff1f; 作品集是个人或公司展示其过往工作成果的集合&#xff0c;通常包括各种专案、作品或成就的范例&#xff0c;用以展示创建者的技能、经验和专业水平。 …

视频改字祝福 豪车装X系统源码uniapp前端源码

内容目录 一、详细介绍二、效果展示1.部分代码2.效果图展示 三、学习资料下载 一、详细介绍 uniapp视频改字祝福 豪车装X系统源码 全开源。 创意无限&#xff01;AI视频改字祝福&#xff0c;豪车装X系统源码开源&#xff0c;打造个性化祝福视频不再难&#xff01; 想要为你的…