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.mzph.cn/web/6276.shtml

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

相关文章

【HAL库 STM32】输入捕获并实现超声波测距

文章目录 HC-SR04 超声波模块简介HC-SR04 工作原理如何使用HC-SR04模块程序效果 一、工程配置代码如果您发现文章有错误请与我留言&#xff0c;感谢 HC-SR04 超声波模块简介 HC-SR04 工作原理 模块有2个超声波换能器&#xff08;如图所示&#xff09;&#xff0c;一个发出声波…

三. Django项目之电商购物商城 -- 校验用户名 , 数据入库

Django项目之电商购物商城 – 校验用户名 , 数据入库 需要开发文档和前端资料的可私聊 一. 路由匹配获得用户名 在注册时 , 用户输入用户名 , 通过ajax请求发送到服务器 , 在路由中设置对应url , 响应视图 , 将用户输入的用户名传入视图 , 与数据库进行校验检查用户名是否重…

LeetCode 110.平衡二叉树(Java/C/Python3/Go实现含注释说明,Easy)

标签 树深度优先搜索递归 题目描述 给定一个二叉树&#xff0c;判断它是否是高度平衡的二叉树。 本题中&#xff0c;一棵高度平衡的二叉树定义为&#xff1a; 一个二叉树每个节点的左右两个子树的高度差的绝对值不超过1。 原题&#xff1a;LeetCode 110.平衡二叉树 思路及…

MFC列表控件用ADO添加数据实例

1、本程序基于前期我的博客文章《MFC用ADO连接ACESS数据库实例(免费源码下载)》 程序功能通过编辑框、组合框实时将数据写入ACESS数据库并在列表控件上显示。 2、在主界面资源视图上加上一个按钮控件、两个静态文本、一个编辑框IDC_EDIT1变量名name、一个组合框IDC_COMBO1变量名…

【Java从入门到精通】Java 流(Stream)、文件(File)和IO

Java.io 包几乎包含了所有操作输入、输出需要的类。所有这些流类代表了输入源和输出目标。 Java.io 包中的流支持很多种格式&#xff0c;比如&#xff1a;基本类型、对象、本地化字符集等等。 一个流可以理解为一个数据的序列。输入流表示从一个源读取数据&#xff0c;输出流…

021、Python+fastapi,第一个Python项目走向第21步:ubuntu 24.04 docker 安装mysql8、redis(二)

系列文章目录 pythonvue3fastapiai 学习_浪淘沙jkp的博客-CSDN博客https://blog.csdn.net/jiangkp/category_12623996.html 前言 安装redis 我会以三种方式安装&#xff0c; 第一、直接最简单安装&#xff0c;适用于测试环境玩玩 第二、conf配置安装 第三、集群环境安装 一…

ASP.NET视频点播系统的设计与实现

摘 要 本文阐述了基于WEB的交互式视频点播系统的协议原理、软件结构和设计实现。本视频点播系统根据流媒体传输原理&#xff0c;在校园局域网的基础上模拟基于Web的视频点播系统&#xff0c;实现用户信息管理、视频文件的添加、删除、修改及在线播放和搜索功能。本系统是一个…

llama-factory/peft微调千问1.5-7b-chat

目标 使用COIG-CQIA数据集和通用sft数据集对qwen1.5-7b-chat进行sft微调,使用公开dpo数据集进行dpo对齐。学习千问的长度外推方法。 一、训练配置 使用Lora方式, 将lora改为full即可使用全量微调。 具体的参数在 该框架将各个参数、训练配置都封装好了,直接使用脚本,将数…

Redis高并发可用-主从复制,集群

Redis高并发可用 1 复制 默认情况下&#xff0c;Redis都是主节点。每个从节点只能有一个主节点&#xff0c;而主节点可以同时具有多个从节点。复制的数据流是单向的&#xff0c;只能由主节点复制到从节点。 1.1 复制的拓扑结构 一主一从&#xff1a; 主一从结构是最简单的…

【全网最全】2024五一数学建模B题前四问代码多种保奖进阶思路+建模过程+代码+数据处理+论文写作技巧等(后续会更新)

您的点赞收藏是我继续更新的最大动力&#xff01; 一定要点击如下的卡片&#xff0c;这是获取资料的入口&#xff01; 【全网最全】2024五一数学建模B题前四问代码多种保奖进阶思路建模过程代码数据处理论文写作技巧等&#xff08;后续会更新&#xff09;「首先来看看目前已有…

LeetCode 94.二叉树的中序遍历

题目描述 给定一个二叉树的根节点 root &#xff0c;返回 它的 中序 遍历 。 示例 1&#xff1a; 输入&#xff1a;root [1,null,2,3] 输出&#xff1a;[1,3,2]示例 2&#xff1a; 输入&#xff1a;root [] 输出&#xff1a;[]示例 3&#xff1a; 输入&#xff1a;root [1] …

R语言学习—1—将数据框中某一列数据改成行名

将数据框中某一列数据改成行名 代码 结果

观察者模式实战:解密最热门的设计模式之一

文章目录 前言一、什么是观察者模式二、Java实现观察者模式2.1 观察者接口2.2 具体观察者2.3 基础发布者2.4 具体发布者2.5 消息发送 三、Spring实现观察者模式3.1 定义事件类3.2 具体观察者3.3 具体发布者3.4 消息发送 总结 前言 随着系统的复杂度变高&#xff0c;我们就会采…

vivado Aurora 8B/10B IP核(9)- CRC、 Aurora 8B/10B内核的时钟接口端口

CRC 模块提供 16 位或 32 位 CRC&#xff0c;用于用户数据。 Aurora 8B/10B 内核的时钟接口端口 从相邻收发器四边形的时钟Xilinx 实现工具可以根据需要对南北路由和引脚交换到收发器时钟输入进行必要的调整&#xff0c;以将时钟从一个四线到另一个。 重要信息&#xff1a;共…

踏上R语言之旅:解锁数据世界的神秘密码(五)

线性与非线性模型及R使用 文章目录 线性与非线性模型及R使用一、数据的分类与模型选择1.变量的取值类型 二、广义线性模型广义线性模型概述Logistic模型 总结 一、数据的分类与模型选择 1.变量的取值类型 因变量记为y&#xff0c;解释变量记为x1&#xff0c;x2,… 因变量y一般…

第三节课,功能2:开发后端用户的管理接口-- postman--debug测试

一、如何使用postman 网址&#xff1a; https://www.postman.com/downloads/ 【Postman小白教程】五分钟学会如何使用Postman~_哔哩哔哩_bilibili postman安装使用_bowser agent在postman哪里-CSDN博客 二、下载后 登录&#xff0c;开始测试 2.1 关于postman 报错&#…

MTEB - Embedding 模型排行榜

文章目录 关于 MTEBMTEB 任务和数据集概览使用 MTEB Pythont 库Installation使用 关于 MTEB MTEB : Massive Text Embedding Benchmark github : https://github.com/embeddings-benchmark/mtebhuggingface : https://huggingface.co/spaces/mteb/leaderboardpaper : https:/…

【Java EE】CAS原理和实现以及JUC中常见的类的使用

˃͈꒵˂͈꒱ write in front ꒰˃͈꒵˂͈꒱ ʕ̯•͡˔•̯᷅ʔ大家好&#xff0c;我是xiaoxie.希望你看完之后,有不足之处请多多谅解&#xff0c;让我们一起共同进步૮₍❀ᴗ͈ . ᴗ͈ აxiaoxieʕ̯•͡˔•̯᷅ʔ—CSDN博客 本文由xiaoxieʕ̯•͡˔•̯᷅ʔ 原创 CSDN 如…

asyncionetworkxFuncAnimation学习--动态显示计算图的运行情况

asyncio&networkx&FuncAnimation学习--动态显示计算图的运行情况 一.效果二.代码 一.目的 1.动态显示计算图的运行状态(点或边是否已完成) 二.步骤: 1.定义计算图 2.asyncio 并行计算 3.networkx 显示计算图 4.FuncAnimation 动态更新 三.依赖: conda install pygraphv…

数据结构--栈与队列【您的关注是我创作的动力!】

文章目录 栈什么是栈&#xff1f;栈的具体实现 队列什么是队列&#xff1f;队列的实现 栈 什么是栈&#xff1f; 栈也是顺序表的一种&#xff0c;栈的逻辑实现是先进后出&#xff08;后进先出&#xff09;就跟子弹夹一样。 具体逻辑就是它只允许在固定的一端进行数据的插入与…