Mybatis四种实例化对象方式

news2024/12/25 13:01:19

代码准备

创建mybatis-config.xml
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE configuration
        PUBLIC "-//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;

    @BeforeEach
    public void before() {
        InputStream inputStream = ConfigurationTest.class.getClassLoader().getResourceAsStream("mybatis-config.xml");
        sqlSessionFactory = new SqlSessionFactoryBuilder().build(inputStream);
    }
}

1.使用默认构造方法

@Test
public void testDefaultConstructor() {
    // 获取sqlSession
    SqlSession 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() {
    // 获取sqlSession
    SqlSession 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() {
    // 获取sqlSession
    SqlSession 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> {

    @Override
    public 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;
        }
    }

    @Override
    public 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;
    }

    @Override
    public 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;
    }

    @Override
    public 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() {
    // 获取sqlSession
    SqlSession 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> {

    @Override
    public 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;
        }
    }

    @Override
    public Additional getNullableResult(ResultSet rs, String columnName) throws SQLException {
        String val = rs.getString(columnName);
        return new Gson().fromJson(val, Additional.class);
    }

    @Override
    public Additional getNullableResult(ResultSet rs, int columnIndex) throws SQLException {
        String val = rs.getString(columnIndex);
        return new Gson().fromJson(val, Additional.class);
    }

    @Override
    public 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() {
    // 获取sqlSession
    SqlSession 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.coloradmin.cn/o/1638984.html

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

相关文章

【golang学习之旅】深入理解字符串string数据类型

系列文章 【golang学习之旅】报错&#xff1a;a declared but not used 【golang学习之旅】Go 的基本数据类型 目录 系列文章使用示例string的底层数据结构关于字符串复制字符串是不可变的如何高效的进行字符串拼接&#xff1f; 使用示例 Go 语言中的字符串只是一个只读的字节…

CUDA CPP Unity Compute Shader

为学 开始一个新的学习计划&#xff0c;涵盖&#xff1a; 主题学习内容CUDAProfessional CUDA C Programming/NVIDIA CUDA初级教程视频(周斌)CCPrimer / The Cherno CPPUnity Compute ShaderUdemy Learn to Write Unity Compute ShadersLinear AlgebraMIT 18.06 Prof.Gilbert…

typescript类型检查和原始类型

typescript类型检查和原始类型 类型检查 非严格类型是typescript默认的类型检查模式&#xff0c;在该模式下&#xff0c;类型检查的规则相对轻松&#xff0c;不会对undefined和null值做过多的限制&#xff0c;允许将undefined和null值赋给string类型的变量。进行JavaScript代…

【算法】高精度乘法

前言 最近在参加某个比赛的时候遇到了这个问题&#xff0c;用字符串表示时&#xff0c;长度能达到15&#xff0c;所以针对大数乘法写一篇文章。 高精度 * 低精度 在这种场景下&#xff0c;一般都是给定一个无法用int或long long 存储的数&#xff0c;再给定一个能用int或lon…

第74天:漏洞发现-Web框架中间件插件BurpSuite浏览器被动主动探针

目录 思维导图 前置知识 案例一&#xff1a;浏览器插件-辅助&资产&漏洞库-Hack-Tools&Fofa_view&Pentestkit 案例二&#xff1a; BurpSuite 插件-被动&特定扫描-Fiora&Fastjson&Shiro&Log4j 思维导图 前置知识 目标&#xff1a; 1. 用…

Linux 进程间通信之命名管道

&#x1f493;博主CSDN主页:麻辣韭菜&#x1f493;   ⏩专栏分类&#xff1a;Linux知识分享⏪   &#x1f69a;代码仓库:Linux代码练习&#x1f69a;   &#x1f339;关注我&#x1faf5;带你学习更多Linux知识   &#x1f51d; 目录 前言 命名管道 创建一个命名管道 …

八大排序详解:动图、代码、注释

目录 何为八大排序&#xff1f; 直接插入排序 排序过程解读 直接插入排序的特性总结&#xff1a; 希尔排序 希尔排序的特性总结&#xff1a; 直接选择排序 直接选择排序的特性总结&#xff1a; 堆排序 直接选择排序的特性总结&#xff1a; 冒泡排序 快速排序 1.Hoa…

全景剖析阿里云容器网络数据链路(七):Terway DataPath V2(Terway≥1.8.0)

作者&#xff1a;余凯 前言 近几年&#xff0c;企业基础设施云原生化的趋势越来越强烈&#xff0c;从最开始的IaaS化到现在的微服务化&#xff0c;客户的颗粒度精细化和可观测性的需求更加强烈。容器网络为了满足客户更高性能和更高的密度&#xff0c;也一直在高速的发展和演…

【JavaEE网络】网络编程及其应用概述

目录 面向字节流粘包问题 TCP异常情况TCP/UDP对比 网络层重点协议IP协议IP地址 面向字节流 粘包问题 在面向字节流的情况下&#xff0c;会产生一些其他的问题&#xff1a;粘包问题&#xff0c;这里“粘”的是“应用层数据报”&#xff0c;通过TCP read/write的数据&#xff0…

jvm 马士兵 01

01.JVM是什么 JVM是一个跨平台的标准 JVM只识别class文件&#xff0c;符合JVM规范的class文件都可以被识别

javaScript 判断闰年

接受用户输入年份 如果是闰年就弹出闰年&#xff0c;遇到平年就是弹出平年 var a prompt(请输入年份);if(a%40&&a%100!0||a%4000){alert(闰年);}else{alert(平年);}

智慧校园为师生带来的那些帮助

随着互联网技术的发展&#xff0c;学校高度重视校园信息化建设&#xff0c;越来越多的学校开始建设智能校园。智慧校园是以智慧校园建设为基础&#xff0c;为学生的校园生活和学校的日常管理带来生机和活力。 那么&#xff0c;在当代环境下建设智慧校园的必要性是什么呢&#x…

Java集合框架-容器源码分析

Java集合框架-容器&源码分析 文章目录 Java集合框架-容器&源码分析[TOC](文章目录)前言一、集合框架概述二、Collection接口及其子接口(List/Set)及实现类2.1 Collection接口中方法2.2 遍历&#xff1a;Iterator迭代器接口&foreach(5.0新特性)2.3 Connection子接口…

Java中使用Redis实现分布式锁的三种方式

1. 导语 随着软件开发领域的不断演进,并发性已经成为一个至关重要的方面,特别是在资源跨多个进程共享的分布式系统中。 在Java中,管理并发性对于确保数据一致性和防止竞态条件至关重要。 Redis作为一个强大的内存数据存储,为在Java应用程序中实现分布式锁提供了一种高效的…

WSL2连接Windows主机的Mysql

文章目录 需求查看主机IP防火墙设置Mysql设置允许远程连接WSL2连接Mysql 需求 在WSL2&#xff08;本机Ubuntu20.04&#xff09;运行的程序需要将数据写入到本机的Mysql服务器中 查看主机IP 两种办法&#xff1a; Windows主机输入 ipconfig&#xff0c;找到带有WSL后缀的部分…

第13章 软件测评相关标准

一、标准化概述 &#xff08;一&#xff09;概念 1、标准 一定范围内获得最佳秩序&#xff0c;经协商一致并由公认机构批准共同使用和重复使用的一种规范性文档&#xff0c;是标准化活动的核心产物。 2、标准化 一定范围内获得最佳秩序&#xff0c;对现实问题和潜在问题制…

ctfshow——SSRF

文章目录 web 351web 352web 353web 354web 355web 356web357web 358web 359web 360 SSRF(Server-Side Request Forgery&#xff1a;服务器端请求伪造) 是一种由攻击者构造形成由服务端发起请求的一个安全漏洞。一般情况下&#xff0c;SSRF攻击的目标是从外网无法访问的内部系统…

Java | Leetcode Java题解之第64题最小路径和

题目&#xff1a; 题解&#xff1a; class Solution {public int minPathSum(int[][] grid) {if (grid null || grid.length 0 || grid[0].length 0) {return 0;}int rows grid.length, columns grid[0].length;int[][] dp new int[rows][columns];dp[0][0] grid[0][0]…

【C/C++基础实战】:用C++实现通讯录管理系统——含完整源码

文章目录 通讯录管理系统一、系统需求以及成品演示二、代码实现三、完整代码 通讯录管理系统 一、系统需求以及成品演示 1.1 系统需求 通讯录是一个可以记录亲人、好友信息的工具。这里利用C来实现一个通讯录管理系统 系统中需要实现的功能如下&#xff1a; 添加联系人&am…

【C语言】/*C语言常见概念*/

目录 前言 一、C语言是什么 二、初识编译和链接 三、什么是可执行程序 四、什么是编译器 五、什么是集成开发环境 六、mian函数的特点 七、什么是关键字 八、标识符的命名规则是什么 九、字符和ASCII码表 十、字符串和\0 十一、转义字符 十二、注释 前言 本篇文章…