第一节:输入参数和输出参数

Mapper.xml映射文件中定义了操作数据库的sql,每个sql是一个statement,映射文件是mybatis的核心。

1.1 环境准备

第一步:创建项目添加依赖

项目名称:mybatis01
依赖信息如下:

<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的核心配置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><!-- 和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日志文件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

User.java

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

第四步:加入sql映射文件

在java目录下创建com.mapper文件夹,然后创建UserMapper.xml文件

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapperPUBLIC "-//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/mapper/UserMapper.xml"/>
</mappers>

1.2 parameterType(输入参数)

1.2.1. 传递简单类型

传递int类型、String类型
使用#{}占位符,或者${}进行sql拼接。 (order by id 必须使用${})

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

<select id="findByWhere" resultType="user">select * from user where username like #{username} and sex=#{sex};
</select>
1.2.2. 传递pojo对象

传递pojo对象
#{}或者${}括号中的值为pojo属性名称。

<insert id="add" parameterType="user">insert into `user` (username,birthday,sex,address) values (#{username},#{birthday},#{sex},#{address})</insert>
1.2.3. 传递pojo包装对象

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

包装对象:Pojo类中的一个属性是另外一个pojo。
需求1:根据用户名模糊查询用户信息

查询条件放到QueryVo的user属性中。

编写QueryVo

public class QueryVo {private User user;private String  username;public User getUser() {return user;}public void setUser(User user) {this.user = user;}/get set
}

使用SQL语句:SELECT * FROM user WHERE username LIKE ‘%枫%’;

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

<select id="findUserByQueryVo" paramenterType="queryVo" resultType="user">select * from user where username like #{user.username}
</select>

测试:
在UserMapper接口中添加方法,如下:

List<User> findUserQueryVo(QueryVo queryVo);

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

@Test
public void testQueryUserByQueryVo() {// MyBatis和spring整合,整合之后,交给spring管理SqlSession sqlSession = MyBatisUtils.openSession();// 创建Mapper接口的动态代理对象,整合之后,交给spring管理UserMapper userMapper = sqlSession.getMapper(UserMapper.class);// 使用userMapper执行查询,使用包装对象QueryVo queryVo = new QueryVo();// 设置user条件User user = new User();user.setUsername("%枫%");// 设置到包装对象中queryVo.setUser(user);// 执行查询List<User> list = userMapper.findUserByQueryVo(queryVo);for (User u : list) {System.out.println(u);}sqlSession.close();
}

查询结果:

这样就可以把数据库中带有“枫”的名字查询出来。

1.2.4. 传递map集合

需求2:分页查询用户信息
分页属性放入map集合中,注意:map的key要和sql中的占位符保持名字一致。

使用SQL语句:SELECT * FROM user limit #{offset},#{pagesize}

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

<!-- 分页:map传参 --><select id="findUserByPage" parameterType="map" resultType="user">SELECT * FROM user LIMIT #{offset}, #{pagesize}</select>

在UserMapper接口中添加方法,如下:

List<User> findUserByPage(Map<String, Object> map);

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

@Test
public void testQueryUserByPage() {SqlSession sqlSession = MyBatisUtils.openSession();UserMapper userMapper = sqlSession.getMapper(UserMapper.class);Map<String, Object> map = new HashMap<String, Object>();map.put("offset", 0);map.put("pagesize", 2);// 执行查询List<User> list = userMapper.findUserByPage(map);for (User u : list) {System.out.println(u);}sqlSession.close();
}

1.3. resultType(输出参数)

1.3.1. 输出简单类型

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

使用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();
}

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

1.3.2. 输出pojo对象(返回user对象,栗子:根据id查询)
1.3.3. 输出pojo列表(返回User对象集合,栗子:查询所有user)

1.4. resultMap

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

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

resultMap可以实现将查询结果映射为复杂类型的pojo,比如在查询结果映射对象中包括pojo和list实现一对一查询和一对多查询。
需求:查询订单表order的所有数据
使用sql语句:SELECT id, user_id, number, createtime, note FROM orders

1.4.1.创建pojo对象

Order对象:

public class Order {// 订单idprivate int id;// 用户idprivate Integer userId;// 订单号private String number;// 订单创建时间private Date createtime;// 备注private String note;//get/set方法
}
1.4.2. 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.mapper.OrderMapper"><!-- 查询所有的订单数据 --><select id="queryOrderAll" resultType="order">SELECT id, user_id,number,createtime, note FROM `order`</select>
</mapper>
1.4.3. Mapper接口

编写接口如下:

public interface OrderMapper {/*** 查询所有订单* * @return*/List<Order> queryOrderAll();
}
1.4.4. 测试方法

编写测试方法OrderMapperTest如下:

@Testpublic void testQueryAll() {// 获取sqlSessionSqlSession sqlSession = MyBatisUtils.openSession();// 获取OrderMapperOrderMapper orderMapper = sqlSession.getMapper(OrderMapper.class);// 执行查询List<Order> list = orderMapper.queryOrderAll();for (Order order : list) {System.out.println(order);}}

发现userId为null
解决方案:使用resultMap

1.4.5. 使用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.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 id, user_id,number,createtime, note FROM `order`</select>
</mapper>

再次测试数据没有问题。

1.5 类型转换器

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

myBatis类型转换器适用于 Java实体类中的类型和数据库中的类型不对应时。
下图是默认的TypeHandler:

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

public class Address {private String address;public Address() {}public Address(String address) {this.address = address;}public String getAddress() {return address;}public void setAddress(String address) {this.address = address;}
}

(2)编写类型转换器

package com.convert;import com.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的类型*          */@Overridepublic 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*/@Overridepublic Address getResult(ResultSet rs, String columnName) throws SQLException {System.out.println(columnName);String value=rs.getString(columnName);Address address=new Address(value);return address;}@Overridepublic Address getResult(ResultSet rs, int columnIndex) throws SQLException {//System.out.println("xxxxxxxxx");return null;}@Overridepublic Address getResult(CallableStatement cs, int columnIndex) throws SQLException {//System.out.println("yyyyyyyyy");return null;}
}

(2)注册类型转换器
在mybatis的核心配置文件中添加

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

这样类型装换器就注册成功了。剩下的内容会在下一篇博客。
Mybatis进阶——高级应用2

MyBatis进阶——高级使用1相关推荐

  1. MyBatis的高级映射之多对一

    第五节 MyBatis的高级映射之多对一 2016年3月4日 星期五 09:25 使用传统方式的形式 使用MyBatis的方式 这样会产生两条语句 使用ResultMap的方式,对结果进行映射和转换, ...

  2. 跟随大咖修炼运营圈求生欲:从案例中学习,进阶高级运营

    作者:波波 运营小咖秀 运营圈的浮躁不可避免. 问题应运而生,解法却莫衷一是.内容.流量.转化--每一个概念都变得立体成为前行的庞大障碍.或许遇到知识短板,或许面对瓶颈无从下手--当你在浮躁中迷失,不 ...

  3. 摆脱运营打杂,进阶高级运营!运营小咖秀特训营第3期圆满落幕

    作者:学运营就来 运营小咖秀 · 懂 · 运 · 营 · 更 · 懂 · 你 · 2019年4月20日-4月21日,由运营小咖秀主办的「摆脱运营打杂,进阶高级运营」线下特训营活动,在北京东城区青龙胡同 ...

  4. 如何才能进阶高级Java程序员?

    身为程序员,一旦进入技术行列,就开启了持续学习的道路,更迭迅速的互联网时代,技术自然也是一代一代的更新,在技术进阶的道路上,要不断吸收新的想法和技术知识.进阶高级Java程序员要怎么做,不如跟着千锋重 ...

  5. 【Java】Mybatis进阶篇(一)

    Mybatis进阶篇 核心配置文件 1.mybatis-config.xml 2.Mybatis 的配置文件包含了会深深影响Mybatis行为的设置和属性信息 Configuration(配置) pr ...

  6. SSM之Mybatis框架高级

    1.resultMap高级操作:联合查询操作 #1 一对一关联查询 例如:根据订单关联查询用户 输出映射: resultType:自定义一个拓展的的pojo resultMap: 这里的resultM ...

  7. Mybatis学习记录(六)----Mybatis的高级映射

    作者:余家小子 1.一对多查询 1.1 需求 查询订单及订单明细的信息. 1.2 sql语句 确定主查询表:订单表 确定关联查询表:订单明细表 在一对一查询基础上添加订单明细表关联即可. SELECT ...

  8. Mybatis(四) 高级映射,一对一,一对多,多对多映射

    天气甚好,怎能不学习? 一.单向和双向 包括一对一,一对多,多对多这三种情况,但是每一种又分为单向和双向,在hibernate中我们就详细解析过这单向和双向是啥意思,在这里,在重复一遍,就拿一对多这种 ...

  9. MyBatis学习--高级映射

    简介 前面说过了简单的数据库查询和管理查询,在开发需求中有一些一对一.一对多和多对多的需求开发,如在开发购物车的时候,订单和用户是一对一,用户和订单是一对多,用户和商品是多对多.这些在Hibernat ...

最新文章

  1. centos7上开启单用户模式
  2. 【BZOJ1899】[Zjoi2004]Lunch 午餐 贪心+DP
  3. 【 Grey Hack 】万金油脚本:路由器漏洞检测
  4. java线程画动图闪,Android中利用画图类和线程画出闪烁的心形,android心形,package com....
  5. @SuppressLint(HandlerLeak),或Handler使用有警告;
  6. 期货交易的异常交易认定标准
  7. 虚拟机设置静态IP地址
  8. 常见Gof设计模式(包括5种创建型、7种结构型、11种行为型)
  9. 微信小程序文字语音转换/中英文自动翻译
  10. 使用vcpkg安装cgal前安装yasm报错
  11. windows安装memcached
  12. OSCP - Typhoon 1.02 的破解
  13. 华为路由器交换机配置命令
  14. 入行程序员培训还是不培训
  15. php 统计汉字,PHP 统计实时统计汉字个数和区别
  16. Samba服务所使用的端口和协议
  17. Bob 的生存概率问题
  18. 通过Python终端输入命令对NAO机器人进行实时控制
  19. postgresql数据库使用Citus实现集群
  20. October cms-Backend (后端-表单)

热门文章

  1. 花生凝集素Peanut Agglutinin(PNA)修饰荧光素FITC、罗丹明、辣根酶HRP、生物素Biotin、Cy7、Cy5、Cy3、Cy5.5、ICG吲哚菁绿
  2. 解密知名NFT收藏家的交易方式
  3. Java C语言 输出n以内的所有素数 以及判断一个数是不是素数
  4. java网络编程-实现聊天程序
  5. 真机联调H5开发的APP
  6. Android实现类似Excel的大表格,可横向纵向滑动,带表头
  7. 多级增益的音频放大器
  8. 什么是ARM 的big.LITTLE
  9. 操作系统--第九章 操作系统接口--习题答案
  10. 选择一个创业项目应该关注哪几个方面