Mapper.xml映射文件中定义了操作数据库的sql,每个sql是一个statement,映射文件是mybatis的核心。
第一步:创建项目添加依赖
项目名称:mybatis01
依赖信息如下:
<dependencies>
<dependency>
<groupId>mysqlgroupId>
<artifactId>mysql-connector-javaartifactId>
<version>5.1.41version>
dependency>
<dependency>
<groupId>org.mybatisgroupId>
<artifactId>mybatisartifactId>
<version>3.4.6version>
dependency>
<dependency>
<groupId>log4jgroupId>
<artifactId>log4jartifactId>
<version>1.2.17version>
dependency>
dependencies>
第二步:加入配置文件
Mybatis的核心配置mybatis-config.xml
<configuration>
<environments default="development">
<environment id="development">
<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文件
<mapper namespace="">
第五步:加载映射文件
MyBatis框架需要加载UserMapper.xml映射文件,将UserMapper.xml添加到mybatis-config.xml,如下:
<mappers>
<mapper resource="com/mapper/UserMapper.xml"/>
mappers>
传递int类型、String类型
使用#{}占位符,或者${}
进行sql拼接。 (order by id 必须使用${})
扩展:如果传递多个简单类型,使用@Param注解实现
List
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>
传递pojo对象
#{}或者${}括号中的值为pojo属性名称。
<insert id="add" parameterType="user">
insert into `user` (username,birthday,sex,address) values (#{username},#{birthday},#{sex},#{address})
insert>
开发中通过可以使用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();
}
需求2:分页查询用户信息
分页属性放入map集合中,注意:map的key要和sql中的占位符保持名字一致。
使用SQL语句:SELECT * FROM user limit #{offset},#{pagesize}
在UserMapper.xml中配置sql,如下。
<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();
}
需求:查询用户表数据条数
使用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();
}
注意:输出简单类型必须查询出来的结果集有一条记录,最终将第一个字段的值转换为输出类型。
resultType可以指定将查询结果映射为pojo,但需要pojo的属性名和sql查询的列名一致方可映射成功。
如果sql查询字段名和pojo的属性名不一致,可以通过resultMap将字段名和属性名作一个对应关系 ,resultMap实质上还需要将查询结果映射到pojo对象中。
resultMap可以实现将查询结果映射为复杂类型的pojo,比如在查询结果映射对象中包括pojo和list实现一对一查询和一对多查询。
需求:查询订单表order的所有数据
使用sql语句:SELECT id, user_id, number, createtime, note FROM orders
Order对象:
public class Order {
// 订单id
private int id;
// 用户id
private Integer userId;
// 订单号
private String number;
// 订单创建时间
private Date createtime;
// 备注
private String note;
//get/set方法
}
创建OrderMapper.xml配置文件
<mapper namespace="com.mapper.OrderMapper">
<select id="queryOrderAll" resultType="order">
SELECT id, user_id,number,createtime, note FROM `order`
select>
mapper>
编写接口如下:
public interface OrderMapper {
/**
* 查询所有订单
*
* @return
*/
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
由于上边的mapper.xml中sql查询列(user_id)和Order类属性(userId)不一致,所以查询结果不能映射到pojo中。需要定义resultMap,resultMap将sql查询列(user_id)和Order类属性(userId)对应起来
改造OrderMapper.xml,如下:
<mapper namespace="com.mapper.OrderMapper">
<resultMap type="order" id="orderResultMap">
<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>
再次测试数据没有问题。
每当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的类型
* */
@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.convert.AddressHandler" javaType="com.pojo.Address" jdbcType="VARCHAR">typeHandler>
typeHandlers>
这样类型装换器就注册成功了。剩下的内容会在下一篇博客。
Mybatis进阶——高级应用2