MyBatis高级应用学习笔记

输入参数和输出参数
输入参数
环境准备

先创建项目,添加依赖

<dependencies>
  		<dependency>
           <groupId>mysql</groupId>
           <artifactId>mysql-connector-java</artifactId>
           <version>5.1.41</version>
       </dependency>
       <!-- https://mvnrepository.com/artifact/org.mybatis/mybatis -->
       <dependency>
           <groupId>org.mybatis</groupId>
           <artifactId>mybatis</artifactId>
           <version>3.4.6</version>
       </dependency>
   <!-- https://mvnrepository.com/artifact/log4j/log4j -->
       <dependency>
           <groupId>log4j</groupId>
           <artifactId>log4j</artifactId>
           <version>1.2.17</version>
       </dependency>
</dependencies>

添加配置文件: Mybatis的核心配置 mybaits-config.xml、log4j日志文件log4j.properties mybaits-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>
	<!-- 和spring整合后 environments配置将废除 -->
	<environments default="development">
		<environment id="development">
			<!-- 使用jdbc事务管理 -->
			<transactionManager type="JDBC" />
			<!-- 数据库连接池 -->
			<dataSource type="POOLED">
				<property name="driver" value="com.mysql.jdbc.Driver" />
				<property name="url"
					value="jdbc:mysql://localhost:3306/mybatis_01?characterEncoding=utf-8" />
				<property name="username" value="root" />
				<property name="password" value="root" />
			</dataSource>
		</environment>
	</environments>
</configuration>

log4j.properties

### set log levels - for more verbose logging change 'info' to 'debug' , 'off' ###
log4j.rootLogger=info, stdout,file

### direct log messages to stdout ###
log4j.appender.stdout=org.apache.log4j.ConsoleAppender
log4j.appender.stdout.Target=System.err
log4j.appender.stdout.layout=org.apache.log4j.PatternLayout
log4j.appender.stdout.layout.ConversionPattern=%d{ABSOLUTE} %5p %c{1}:%L - %m%n

### direct messages to file mylog.log ###
log4j.appender.file=org.apache.log4j.FileAppender
log4j.appender.file.File=d:\\mylog.log
log4j.appender.file.layout=org.apache.log4j.PatternLayout
log4j.appender.file.layout.ConversionPattern=%d{ABSOLUTE} %5p %c{1}:%L - %m%n

创建POJO(Plain Old Java Objects),即实体类,作用和之前的domain一样 >POJO是Plain Old Java Objects的缩写,POJO实质上可以理解为简单的实体类,顾名思义POJO类的作用是方便程序员使用数据库中的数据表,对于广大的程序员,可以很方便的将POJO类当做对象来进行使用,当然也是可以方便的调用其get,set方法。

User.java

public class User {
	private int id;
	private String username;// 用户姓名
	private String sex;// 性别
	private Date birthday;// 生日
	private String address;// 地址
	//getter和setter
}

加入Sql映射文件 在resources目录下创建con.qf.mapper文件夹,然后创建UserMapper.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">
<!-- namespace:命名空间,用于隔离sql-->
<mapper namespace="">
  
</mapper

加载映射文件
MyBatis框架需要加载UserMapper.xml映射文件,将UserMapper.xml添加在mybatis-config.xml,如下:

<mappers>
        <mapper resource="com/qf/mapper/UserMapper.xml"/>
</mappers>
(1) 简单类型

String 基本类型 包装类 #{任意} ${value},可以传递多个简单类型param1 param2 @Param("")注解可以用来给参数传值,使用方法在后面有介绍

传递int类型、String类型
使用#{}占位符,或者KaTeX parse error: Expected 'EOF', got '#' at position 103: … USER where id=#̲{id} </sele…{})

扩展:如果传递多个简单类型,使用@Param注解实现
List findByWhere(@Param(“username”) String username, @Param(“sex”) String sex);
SQL映射文件如下,不需要写parameterType

select * from user where username like #{username} and sex=#{sex};

工具类MyBaitsUtils代码

public class MyBaitsUtils {
    private static SqlSessionFactory factory;
    private static Logger logger;
    static {
        try {
            //1. 获取myBaits配置文件信息
            logger = Logger.getLogger(MyBaitsUtils.class.getName());
            //Resources是import org.apache.ibatis.io.Resources里面的,其他的也都是org.apache.ibatis
            InputStream is = Resources.getResourceAsStream("mybatis-config.xml");
            //2.创建工厂
            factory = new SqlSessionFactoryBuilder().build(is);
        } catch (IOException e) {
            e.printStackTrace();
            logger.error("工厂初始化失败");
        }
    }
    public static SqlSession openSession(){
        return factory.openSession();
    }
}

传递多个参数查找 UserMapper接口中的方法如下:

//这里是用注释的方法,在xml文件中可以使用注释里面的名字
List<User> findByUserNameAndSex(@Param("username") String username, @Param("sex") String sex);

UserMapper.xml中的代码:

<!--根据用户名和性别查询-->
    <select id="findByUserNameAndSex" resultType="User">
        <include refid="selectuser"></include>
        username like #{username} and sex=#{sex}
    </select>

测试代码如下:

 @Test
    public void testByUserNameAndSex(){
        SqlSession sqlSession = MyBaitsUtils.openSession();
        UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
        List<User> all = userMapper.findByUserNameAndSex("%张%","1");
        for (User user : all) {
            System.out.println(user.toString());
        }
        sqlSession.close();
    }
传递pojo对象

传递pojo对象#{}或者${}括号中的值为pojo属性名称。
UserMapper接口中的代码:

void add(User user);

在mybatis-config.xml中添加typeAliases标签

 <!--给所有这个包里面的元素都赋予这个别名-->
 <!--typeAliases中可以用typeAlias标签和package标签,用法相似,typeAlias是给一个类起别名,package是给一个包里面所有类都赋予此别名-->
    <typeAliases>
        <package name="com.qf.pojo"/>
    </typeAliases>
 <!--添加-->
    <insert id="add" parameterType="User">
        insert into user (username,biethday,sex,address) values (#{username},#{birthday},#{sex},#{address})
    </insert>
传递pojo包装对象

开发中通过可以使用pojo传递查询条件。
查询条件可能是综合的查询条件,不仅包括用户查询条件还包括其它的查询条件(比如查询用户信息的时候,将用户购买商品信息也作为查询条件),这时可以使用包装对象传递输入参数。

包装对象:Pojo类中的一个属性是另外一个pojo。
首先要在Pojo包中添加一个QueryVo类,类中添加一个类型是User 的user属性,并添加getXxx()和setXxx()方法
查找姓名包含张,且性别为男的User
然后在UserMapper接口中添加方法:

//传递包装的pojo
    List<User> findByQueryVo(QueryVo queryVo);
<sql id="selectuser">
        select * from user
    </sql>
    
<!--传递包装的pojo (QueryVo)-->
<!--include where是动态sql用法,后面有介绍-->
    <select id="findByQueryVo" resultType="User">
        <include refid="selectuser"></include>
        <where>
            <if test="user.username!=null and user.username!=''">
                and username like #{user.username}
            </if>
            <if test="user.sex!=null and user.sex!=''">
                and sex=#{user.sex}
            </if>
        </where>

    </select>

测试类中新增测试方法:

@Test
    public void testFindByQueryVo(){
        SqlSession sqlSession = MyBaitsUtils.openSession();
        UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
        QueryVo queryVo = new QueryVo();
        queryVo.setUser(new User(0,"%张%","1",null,null));
        List<User> all = userMapper.findByQueryVo(queryVo);
        for (User user : all) {
            System.out.println(user.toString());
        }
        sqlSession.close();
    }

传递map集合

分页属性放入map集合中,注意:map的key要和sql中的占位符保持名字一致。

使用SQL语句:SELECT * FROM user limit #{offset},#{pagesize}
必须使用${}的情况,分页

 <!--分页查询-->
    <select id="findByPage" resultType="User">
        select * from user ${orderby} limit #{offset},#{pagesize}
    </select>

UserMapper接口中的代码:

List<User> findByPage(HashMap<String,Object> map);

测试代码:

@Test
    public void testFindByPage(){
        SqlSession sqlSession = MyBaitsUtils.openSession();
        UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
        HashMap<String,Object> map = new HashMap<>();
        map.put("orderby","order by id");
        map.put("offset",0);
        map.put("pagesize",5);
        List<User> all = userMapper.findByPage(map);
        for (User user : all) {
            System.out.println(user.toString());
        }
        sqlSession.close();
    }
输出参数resultType(输出参数)
输出简单类型

需求:查询用户表数据条数

使用sql:SELECT count(*) FROM user

在UserMapper.xml中配置sql,如下:

 <select id="queryUserCount"  resultType="int">
         SELECT count(*) FROM user
 </select>

在UserMapper添加方法,如下:

int queryUserCount();

在UserMapeprTest增加测试方法,如下:

@Test
public void testQueryUserCount() {
	SqlSession sqlSession = MyBatisUtils.openSession();
	
	UserMapper userMapper = sqlSession.getMapper(UserMapper.class);
	// 使用userMapper执行查询用户数据条数
	int count = userMapper.queryUserCount();
	System.out.println(count);

	sqlSession.close();
}

注意:输出简单类型必须查询出来的结果集有一条记录,最终将第一个字段的值转换为输出类型。

resultMap

resultType可以指定将查询结果映射为pojo,蛋需要pojo属性名和sq查询的列名一致方可映射成功。

如果sql查询字段名和pojo的属性名不一致,可以通过resultMap将字段名和属性名做一个对应关系,resultMap实质上还需要将查询结果映射到pojo对象中。

resultMap可以实现将查询结果映射为复杂类型的pojo,比如在查询结构映射对象中包括pojo和list
实现一对一查询和一对多查询。

需求:查询订单表order的所有数据

实体类order中用户id用的是userId,在数据库中用的是user_id,所以需要用resultMap来一一对应下

使用sql语句:select * from orders

创建pojo对象
Order对象

public class Order {
    // 订单id
    private int id;
    // 用户id    
    private Integer userId;    
    // 订单号    
    private String number;    
    // 订单创建时间    
    private Date createtime;    
    // 备注    
    private String note;    
    //get/set。。。
 }

Mapper.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">
<!-- namespace:命名空间,用于隔离sql,还有一个很重要的作用,Mapper动态代理开发的时候使用,需要指定Mapper的类路径 -->
<mapper namespace="com.qf.mapper.OrderMapper">
	<!-- 查询所有的订单数据 -->
	<select id="queryOrderAll" resultType="order">
		SELECT id, user_id,number,createtime, note FROM `order`
	</select>
</mapper>

OrderMapper接口:

public interface OrderMapper {
	// 查询所有订单 
	List<Order> queryOrderAll();
}

编写测试方法OrderMapperTest如下:

@Test
	public void testQueryAll() {
		// 获取sqlSession
		SqlSession sqlSession = MyBatisUtils.openSession();
		// 获取OrderMapper
		OrderMapper orderMapper = sqlSession.getMapper(OrderMapper.class);

		// 执行查询
		List<Order> list = orderMapper.queryOrderAll();
		for (Order order : list) {
			System.out.println(order);
		}
	}

发现userId为null

解决方案:使用resultMap

使用resultMap

由于上边的mapper.xml中sql查询列(user_id)和Order类属性(userId)不一致,所以查询结果不能映射到pojo中。需要定义resultMap,resultMap将sql查询列(user_id)和Order类属性(userId)对应起来
改造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">
<!-- namespace:命名空间,用于隔离sql,还有一个很重要的作用,Mapper动态代理开发的时候使用,需要指定Mapper的类路径 -->
<mapper namespace="com.qf.mapper.OrderMapper">

	<!-- resultMap最终还是要将结果映射到pojo上,type就是指定映射到哪一个pojo -->
	<!-- id:设置ResultMap的id -->
	<resultMap type="order" id="orderResultMap">
		<!-- 定义主键 ,非常重要。如果是多个字段,则定义多个id -->
		<!-- property:主键在pojo中的属性名 -->
		<!-- column:主键在数据库中的列名 -->
		<id property="id" column="id" />

		<!-- 定义普通属性 -->
		<result property="userId" column="user_id" />
		<result property="number" column="number" />
		<result property="createtime" column="createtime" />
		<result property="note" column="note" />
	</resultMap>

	<!-- 查询所有的订单数据 -->
	<select id="queryOrderAll" resultMap="orderResultMap">
		SELECT * FROM `order`
	</select>

</mapper>

再次测试数据没有问题。

类型转换器

每当MyBatis设置参数到PrepareStatement或者从ResultSet结果集中取值时,就会用到TypeHandler来处理数据库类型与Java类型之间的转换。

myBatis类型转换器适用于 Java实体类中的类型和数据库中的类型不对应时。

下图是默认的TypeHandler:
在这里插入图片描述

案例:假如User中包含一个对象属性Address,如果把数据库中address转成Address对象呢?

核心,继承TypeHandler类,加泛型

(1)编写类型转换器

package com.qf.convert;

import com.qf.pojo.Address;
import org.apache.ibatis.type.JdbcType;
import org.apache.ibatis.type.TypeHandler;

import java.sql.CallableStatement;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;

/**
 * wgy 2019/7/6 7:20
 */
public class AddressHandler implements TypeHandler<Address> {
    /**
     * /**
     * 	 * 此方法是在插入是进行设置参数
     * 	 * 参数:	PreparedStatement
     * 	 * 		int	i				为Jdbc预编译时设置参数的索引值
     * 	 * 		Address parameter			要插入的参数值
     * 	 * 		JdbcType jdbcType	要插入JDBC的类型
     *          */
    @Override
    public void setParameter(PreparedStatement ps, int i, Address parameter, JdbcType jdbcType) throws SQLException {
        if(parameter==null){
            ps.setString(i, null);
        }else{
            ps.setString(i, parameter.getAdd());
        }

    }

    /**
     * 该方法是获取参数时执行
     * 参数:	ResultSet rs		查询当前列数据
     * 	 *			String cloumnName	查询当前列名称
     * @param rs
     * @param columnName
     * @return
     * @throws SQLException
     */
    @Override
    public Address getResult(ResultSet rs, String columnName) throws SQLException {
        System.out.println(columnName);
        String value=rs.getString(columnName);
        Address address=new Address(value);
        return address;
    }

    @Override
    public Address getResult(ResultSet rs, int columnIndex) throws SQLException {
        System.out.println("xxxxxxxxx");
        return null;
    }

    @Override
    public Address getResult(CallableStatement cs, int columnIndex) throws SQLException {
        System.out.println("yyyyyyyyy");
        return null;
    }
}

(2)注册类型转换器

在mybatis的核心配置文件中添加

  <!--全局注册类型转换器-->
   <typeHandlers>
        <typeHandler handler="com.qf.convert.AddressHandler" javaType="com.qf.pojo.Address" jdbcType="VARCHAR"></typeHandler>
    </typeHandlers>

(3)测试
只要是查询操作,且查询的结果包含“address”字段就会调用getResult方法,只要是插入操作,且插入的对象含有“address”字段,就会调用setParameter。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值