Mybatis四种实例化对象方式

代码准备

创建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中未指定了构造方法
  • 3
    点赞
  • 10
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Spring MVC是基于Spring框架的一种Web框架,用于开发Java Web应用程序。而Spring是一个轻量级的IoC(Inverse of Control)和AOP(Aspect-Oriented Programming)容器,通过IOC容器,可以管理各种Java Bean的生命周期和依赖关系,而通过AOP可以实现一些跨越多个模块或对象的通用功能。 MyBatis是一个持久层框架,它可以将数据库操作抽象为Java接口的方法,通过配置文件将这些方法映射到相应的SQL语句,实现轻量级的数据库访问。 SpringMVC、Spring和MyBatis可以结合使用来开发一个完整的Java Web项目。 在一个典型的SpringMVC、Spring和MyBatis项目中,通常会有以下几个主要组件: 1. 控制器(Controller):负责接收用户的请求,处理业务逻辑,并返回相应的结果。SpringMVC提供了很多注解和接口,可以很方便地定义和处理请求。 2. 服务(Service):负责处理具体的业务逻辑。通常一个服务类会对应一个或多个DAO(Data Access Object),通过DAO对象访问数据库,获取或修改数据。 3. DAO(Data Access Object):负责封装与数据库交互的细节。通过使用MyBatis的映射文件,将Java接口的方法映射到相应的SQL语句。 4. 数据库:使用关系型数据库(如MySQL、Oracle等)存储持久化数据。 在项目中,通过配置文件和注解,将这些组件关联起来。Spring框架负责管理各个组件的生命周期和依赖关系,MyBatis提供数据访问功能,SpringMVC负责接收用户请求并将结果返回给用户。 通过SpringMVC、Spring和MyBatis的结合,可以实现一个完整的Java Web项目,使开发者能够更加方便、高效地进行开发,并且能够更好地实现各个组件之间的解耦和复用。 ### 回答2: springmvc spring mybatis是一种常见的Java Web开发框架组合,被广泛应用于企业级的Web应用程序开发中。下面以一个简单的项目实例来介绍其基本用法。 假设我们要开发一个简单的学生信息管理系统,使用springmvc来搭建前端控制器,spring来进行依赖注入和事务管理,mybatis作为持久层框架与数据库交互。 首先,我们需要进行项目的初始化配置。在Spring配置文件中,配置相关的bean以及包扫描路径,使得框架能够自动扫描并管理相应的组件。 接下来,我们创建相应的实体类,例如学生类,可以定义属性如姓名、年龄等,并使用注解进行实体类与数据库表的映射。同时,我们可以创建DAO接口和Mapper映射文件,用于扩展mybatis的CURD操作。 然后,我们在Controller中编写处理请求的方法,通过注解标识URL映射关系、请求类型,使用Service层的方法进行业务处理,并将结果传递给视图解析器进行渲染。 在Service层,我们可以编写相应的逻辑代码,例如添加、删除、修改学生信息等。同时,我们可以利用@Autowired注解将DAO层的方法注入到Service中,从而将数据库操作与业务逻辑进行分离。 最后,我们可以在视图层创建相应的页面,例如添加、删除、修改学生信息的页面。利用EL表达式和JSTL标签库等,实现数据的展示和交互。 总的来说,springmvc spring mybatis的项目实例中,我们通过配置框架的相关组件,编写实体类、映射文件以及相应的控制器、服务层和视图层代码,从而实现了一个简单的学生信息管理系统。通过这个实例,我们可以更好地理解和掌握springmvc spring mybatis的基本用法和项目架构。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值