Mybatis

1,MyBatis简介

1.1 MyBatis历史

MyBatis最初是Apache的一个开源项目iBatis, 2010年6月这个项目由Apache Software Foundation迁移到了Google Code。随着开发团队转投Google Code旗下,iBatis3.x正式更名为MyBatis。代码于2013年11月迁移到Github

1.2 MyBatis特性

1) MyBatis是支持定制化SQL、存储过程以及高级映射的优秀的持久层框架

2) MyBatis避免了几乎所有的JDBC代码和手动设置参数以及获取结果集

3) MyBatis可以使用简单的XML或注解用于配置和原始映射,将接口和Java的POJO (Plain Old JavaObjects,普通的Java对象)映射成数据库中的记录

4)MyBatis是一个半自动的ORM (Object Relation Mapping)框架

1.3 MyBatis下载

1.4 和其它持久化层技术对比

1.4.1 JDBC

SQL夹杂在Java代码中耦合度高,导致硬编码内伤

维护不易且实际开发需求中SQL有变化,频繁修改的情况多见

代码冗长,开发效率低

1.4.2 Hibernate和JPA

操作简便,开发效率高

程序中的长难复杂SQL需要绕过框架

内部自动生产的SQL,不容易做特殊优化

基于全映射的全自动框架,大量字段的POJO进行部分映射时比较困难

反射操作太多,导致数据库性能下降

1.4.3 MyBatis

轻量级,性能出色

SQL和Java编码分开,功能边界清晰。Java代码专注业务、SQL语句专注数据

开发效率稍逊于Hlbernate,但是完全能够接受

2,MyBatis的搭建

2.1 开发环境

2.2 创建maven工程

打包方式:jar

引入依赖:

<packaging>jar</packaging>


<dependencies>
    <!-- Mybatis核心 -->
    <dependency>
        <groupId>org.mybatis</groupId>
        <artifactId>mybatis</artifactId>
        <version>3.5.7</version>
    </dependency>
    <!-- junit测试 -->
    <dependency>
    <groupId>junit</groupId>
        <artifactId>junit</artifactId>
        <version>4.12</version>
        <scope>test</scope>
    </dependency>
        <!-- MySQL驱动 -->
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>5.1.47</version>
        </dependency>
    </dependencies>

2.3 创建MyBatis的核心配置文件

核心配置文件主要用于配置连接数据库的环境以及 MyBatis 的全局配置信息
<?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>

    <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:///ssm?useSSL=false"/>
                <property name="username" value="root"/>
                <property name="password" value="111111"/>
            </dataSource>
        </environment>
    </environments>

    <!--mapper映射-->
    <mappers>
        <!--扫描mapper-->
        <package name="org.example.mapper"/>
    </mappers>
</configuration>

2.4 创建mapper接口

public interface UserMapper {
    /**
     * 添加用户信息
     */
    int insertUser();
}

2.5 创建MyBatis的映射文件

ORM:(Object Relationship Mapping)对象关系映射

对象:Java的实体类对象

关系:关系型数据库

映射:二者之间的对应关系

Java概念数据库概念
属性字段/列
对象记录/行

1、映射文件的命名规则:

表所对应的实体类的类名+Mapper.xml

因此一个映射文件对应一个实体类,对应一张表的操作

2、MyBatis中可以面向接口操作数据,要保证两个一致:

1  mapper接口的全类名和映射文件的命名空间(namespace)保持─致

2  mapper接口中方法的方法名和映射文件中编写SQL的标签的id属性保持─致

<?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.atguigu.mybatis.mapper.UserMapper">


    <insert id="insertUser">
        insert into t_user values (null,'admin','123456',18,'男','12345@qq.com')
    </insert>

</mapper>

2.6 通过junit测试功能

//获取核心配置文件的输入流
InputStream is = Resources.getResourceAsStream("mybatis-config.xml");

//获取SqlSessionFactoryBuilder对象
SqlSessionFactoryBuilder sqlSessionFactoryBuilder = new SqlSessionFactoryBuilder();

//获取SqlSessionFactory对象
SqlSessionFactory sqlSessionFactory = sqlSessionFactoryBuilder.build(is);

//获取sql的会话对象SqlSession,是MyBatis提供的操作数据库的对象
SqlSession sqlSession = sqlSessionFactory.openSession();

//获取userMapper的代理实现类对象
UserMapper mapper = sqlSession.getMapper(UserMapper.class);

//调用mapper接口中的方法,实现添加用户信息的功能
int result = mapper.insertUser();
System.out.println("结果为:"+result);

//提交事务
sqlSession.commit();

//释放资源
sqlSession.close();

其中调用mapper方法可以被替换:

//提供sql的唯一标识找到sql并执行
//唯一标识是namespace.sqlId
sqlSqssion.insert("com.atguigu.mybatis.mapper.UserMapper.insertUser);

namespace:
<mapper namespace="com.atguigu.mybatis.mapper.UserMapper">

 因为提交事务比较麻烦,故可以设置自动提交

//获取sql的会话对象SqlSession(不可自动提交),是MyBatis提供的操作数据库的对象
SqlSession sqlSession = sqlSessionFactory.openSession();

//可自动提交
SqlSession sqlSession = sqlSessionFactory.openSession(true);

2.7 加入log4j日志功能

2.7.1 引入依赖

    <!-- log4j日志 -->
        <dependency>
            <groupId>log4j</groupId>
            <artifactId>log4j</artifactId>
            <version>1.2.17</version>
        </dependency>

2.7.2 加入配置文件

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE log4j:configuration SYSTEM "log4j.dtd">
<log4j:configuration xmlns:log4j="http://jakarta.apache.org/log4j/">
    <appender name="STDOUT" class="org.apache.log4j.ConsoleAppender">
        <param name="Encoding" value="UTF-8" />
        <layout class="org.apache.log4j.PatternLayout">
            <param name="ConversionPattern" value="%-5p %d{MM-dd HH:mm:ss,SSS}
%m (%F:%L) \n" />
        </layout>
    </appender>
    <logger name="java.sql">
        <level value="debug" />
    </logger>
    <logger name="org.apache.ibatis">
        <level value="info" />
    </logger>
    <root>
        <level value="debug" />
        <appender-ref ref="STDOUT" />
    </root>
</log4j:configuration>

日志级别:

FATAL(致命)>ERROR(错误)>WARN(警告)>INFO(信息)>DEBUG(调试)

从左到右打印内容越来越详细

打印的信息级别为大于等于该信息级别

如:debug级别会打印所有的级别

2.8 封装SqlSession对象

    public static SqlSession getSqlSession() {
        SqlSession sqlSession =null;
        try {
            //获取核心配置文件的输入流
            InputStream is = Resources.getResourceAsStream("mybatis-config.xml");

            //获取SqlSessionFactoryBuilder对象
            SqlSessionFactoryBuilder sqlSessionFactoryBuilder = new SqlSessionFactoryBuilder();

            //获取SqlSessionFactory对象
            SqlSessionFactory sqlSessionFactory = sqlSessionFactoryBuilder.build(is);

            //获取sql的会话对象SqlSession,是MyBatis提供的操作数据库的对象
            sqlSession = sqlSessionFactory.openSession(true);
        } catch (IOException e) {
            throw new RuntimeException(e);
        }
        return sqlSession;
    }

2.9 映射

resultType:设置结果类型,即查询的数据要转换为的java类型

resultMap:自定义映射,处理多对一或者一对多的映射关系

二者只能设置一个

<select id="getUserById" resultType="com.atguigu.mybatis.pojo.User">

2.10 test

@Test
public void testGetAllUser(){
    SqlSession sqlSession = SqlSessionUtil.getSqlSession();

    UserMapper mapper = sqlSession.getMapper(UserMapper.class);

    List<User> list = mapper.getAllUser();

    list.forEach(System.out::println);

    sqlSession.close();
}

3,核心配置文件

3.1 environments

    <!--
        environments:配置数据库连接环境
        属性:
        default:设置默认使用的环境的id
        如 development test两个environment
        default可选择一个,去使用环境
    -->
    <environments default="development">
        <!--
            environment:配置一个具体的数据库连接环境
            属性:
            id:不能重复
        -->
        <environment id="development">
            <!--
                transactionManager:设置事务管理器
                属性:
                type:设置事务管理的方式(JDBC/MANAGED)
                JDBC:表示使用JDBC中原生的事务管理方式
                MANAGED:被管理,例如:spring
            -->
            <transactionManager type="JDBC"/>
            <!--
                dataSource:设置数据源
                属性:
                type:设置数据库类型(POOLED/UNPOOLED/JNDI)
                POOLED:表示使用数据库连接池
                UNPOOLED:不表示使用数据库连接池
                JNDI:表示使用上下文中的数据源
            -->
            <dataSource type="POOLED">
                <property name="driver" value="com.mysql.jdbc.Driver"/>
                <property name="url" value="jdbc:mysql:///ssm?useSSL=false"/>
                <property name="username" value="root"/>
                <property name="password" value="111111"/>
            </dataSource>
        </environment>

3.2 properties

jdbc.driver=com.mysql.jdbc.Driver
jdbc.url=jdbc:mysql:///ssm?useSSL=false
jdbc.username=root
jdbc.password=111111
<!--引入properties文件,此后就可以在当前文件中使用${key}的方式访问value-->
<properties resource="jdbc.properties"/>

<environment id="test">
    <transactionManager type="JDBC"/>
    <dataSource type="POOLED">
        <property name="driver" value="${jdbc.driver}"/>
        <property name="url" value="${jdbc.url}"/>
        <property name="username" value="${jdbc.username}"/>
        <property name="password" value="${jdbc.password}"/>
    </dataSource>
</environment>

3.3 typeAliases

核心配置文件中的标签必须按照固定的顺序:

properties?,settings?,typeAliases?,typeHandlers?,objectFactory?,objectWrapperFactory?,reflectorFactory?,plugins?,environments?,databaseIdProvider?,mappers?

    <!--
        typeAliases:设置类型别名,即为某个具体的类型设置一个别名
        在MyBatis的范围里,就可以使用别名表示一个具体的类型
    -->
    <typeAliases>
        <!--
            type:设置需要起别名的类型
            alias:设置某个类型的别名
        -->
        <typeAlias type="com.atguigu.mybatis.pojo.User" alias="abc"></typeAlias>
    </typeAliases>
    <select id="getAllUser" resultType="abc">
        select * from t_user
    </select>
<!--若不设置alias,当前类型拥有默认的别名,类名不区分大小写-->
<typeAlias type="com.atguigu.mybatis.pojo.User"></typeAlias>
<!--通过设置类型别名,指定包下所有的类型将会全部拥有默认的别名,类名不区分大小写-->
<package name="com.atguigu.mybatis.pojo"/>

3.4 mappers

    <mappers>
<!--        <mapper resource="mappers/UserMapper.xml"/>-->
        <!--
            以包的方式引入映射文件,但是必须满足两个条件
            1.mapper接口和映射文件所在的包必须一致
            2.mapper接口的名字和映射文件的名字必须一致
        -->
        <package name="com.atguigu.mybatis.mapper"/>
    </mappers>

4,MyBatis获取参数值的两种方式

MyBatis获取参数值的两种方式:${}和#{}

${}的本质就是字符串拼接,#{}的本质就是占位符赋值

${}使用字符串拼接的方式拼接sql,若为字符串类型或日期类型的字段进行赋值时,需要手动加单引号;但是#{}使用占位符赋值的方式拼接sql,此时为字符串类型或日期类型的字段进行赋值时,可以自动添加单引号

4.1 单个字面量类型的参数

若mapper接口中的方法参数为单个的字面量类型

此时可以使用${}和#{}以任意的名称获取参数的值,注意${}需要手动加单引号

select * from t_user where username = #{username}//传入数据zs
//select * from t_user where username = ?


select * from t_user where username = '${username}'//传入数据zs
//select * from t_user where username = 'zs'
如不加''或者"",则为zs,会报错

4.2 多个字面量类型的参数

若mapper接口中的方法参数为多个时,此时MyBatis会自动将这些参数放在一个map集合中,

以arg0,arg1...为键,以参数为值;

以param1,param2...为键,以参数为值;

因此只需要通过${}和#{}访问map集合的键就可以获取相对应的值,注意${}需要手动加单引号

select * from t_user where username = #{username} and password = #{password}
//会报错

select * from t_user where username = #{arg0} and password = #{arg1}

select * from t_user where username = '${param1}' and password = '${param2}'

User checkLogin(String username,String password);

4.3 map集合类型的参数

若mapper接口中的方法需要的参数为多个时,此时可以手动创建map集合,将这些数据放在map中

只需要通过$和#0访问map集合的键就可以获取相对应的值,注意${}需要手动加单引号

select * from t_user where username = #{username} and password = #{password}

User checkLoginByMap(Map<String, Object> map);
Map<String,Object> map = new HashMap<>();
map.put("username","zs");
map.put("password","123");

User zs = mapper.checkLoginByMap(map);

4.4 实体类类型的参数

若mapper接口中的方法参数为实体类对象时 

此时可以使用${}和#{},通过访问实体类对象中的属性名获取属性值,注意${}需要手动加单引号

insert into t_user values (null,#{username},#{password},#{age},#{gender},#{email})

相当于insert into t_user values (null,?,?,?,?,?)

4.5 使用@param标识参数

可以通过@Param注解标识mapper接口中的方法参数

此时,会将这些参数放在map集合中,以

@Param注解的value属性值为键,以参数为值;

以param1,param2...为键,以参数为值;

只需要通过${}和#{}访问map集合的键就可以获取相对应的值

注意${}需要手动加单引号

User checkLoginByParam(@Param("username") String username,@Param("password") String password);

select * from t_user where username = #{username} and password = #{password}

5,MyBatis的查询功能

5.1 查询实体类对象

当查询的数据为多条时,不能使用实体类作为返回值,否则会抛出异常TooManyResultsException

User getUserById(@Param("id") Integer id);

select * from t_user where id = #{id}

5.2 查询list集合

但是若查询的数据只有一条,可以使用实体类或集合作为返回值

List<User> getAllUser();

select * from t_user


List<User> allUser = mapper.getAllUser();
allUser.forEach(System.out::println);

5.3 查询单个数据

在MyBatis中,对于Java中常用的类型都设置了类型别名

例如:java.lang.Integer-->int  /  integer

例如:int-->_int  /  _integer

例如:Map-->map,List-->list

int totalCount();

<select id="totalCount" resultType="Integer">
    select count(*) from t_user
</select>

5.4 查询一条数据为map集合

Map<String, Object> getUserToMap(@Param("id") int id);

<select id="getUserToMap" resultType="Map">
    select * from t_user where id = #{id}
</select>
Map<String, Object> userByIdToMap = mapper.getUserToMap(5);
System.out.println(userByIdToMap);//{password=123456, gender=男, id=5, age=18, email=12345@qq.com, username=admin}

当有数据为空时,则不会显示出来

5.5 查询多条数据为map集合

5.5.1 方法1

将表中的数据以 map 集合的方式查询,一条数据对应一个 map ;若有多条数据,就会产生多个 map 集合,此 时可以将这些map 放在一个 list 集合中获取
List<Map<String,Object>> getAllMap();

List<Map<String, Object>> allMap = mapper.getAllMap();
allMap.forEach(System.out::println);

5.5.2 方法2

将表中的数据以 map 集合的方式查询,一条数据对应一个 map ;若有多条数据,就会产生多个 map 集合,并且最终要以一个map 的方式返回数据,此时需要通过 @MapKey 注解设置 map 集合的键,值是每条数据所对应的map集合
//利用MapKey中的数据作为map的键
@MapKey("id")
Map<String,Object> getAllMap();

Map<String, Object> allMap = mapper.getAllMap();

6,特殊SQL的执行

6.1 模糊查询

List<User> getUserByLike(@Param("username") String username);

//方法1
select * from t_user where username like '%${username}%'
//方法2
select * from t_user where username like concat('%',#{username},'%')
//方法3,方法3不可以使用单引号''
select * from t_user where username like "%"#{username}"%"

6.2 批量删除

void deleteMoreUser(@Param("ids") String ids);

delete from t_user where id in (${ids})
//不可以使用#{},因为会自动增加'',使数据变为in ('7,8'),从而错误

mapper.deleteMoreUser("7,8");

6.3 动态设置表名

List<User> getUserList(@Param("tableName") String tableName);

select * from ${tableName}
//不可使用#{},#{}会自动加'',从而报错

6.4 添加功能获取自增的主键

useGeneratedKeys:设置使用自增的主键

keyProperty: 因为增删改有统一的返回值是受影响的行数,因此只能将获取的自增的主键放在传输的参 user 对象的某个属性中
void insertUser(User user);

<insert id="insertUser" useGeneratedKeys="true" keyProperty="id">
    insert into t_user values (null,#{username},#{password},#{age},#{gender},#{email})
</insert>

  • 10
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值