分享

Mybatis学习笔记1

 太极混元天尊 2018-04-17


Java学习
针对Java初学者以及自学者!
公告通知

如果你是初学者,或者是自学者!你可以加小编微信(xxf960513)!小编可以给你学习上,工作上的一些建议以及可以给你(免费)提供学习资料!最重要我们还可以交个朋友!你在学习上有什么问题都可以加小编微信进行私聊!小编都会为你解答!

注:本公众号纯属个人公益公众号!不存在任何收费项目!免费分享所有学习资料!只求大家多多支持!多多关注!!

本章目录
Mybatis学习笔记1

1、mybatis的介绍

2、使用jdbc操作数据库存在的问题

3、 Mybatis的架构

4、Mybatis的入门程序

   4.1根据用户id查询一个用户信息

   4.2根据用户名称模糊查询用户信息列表

   4.3添加用户

   4.4更新用户

   4.5删除用户


提示:点击文章底部的“阅读原文”查看以前所有的笔记
1、MyBatis介绍

MyBatis 本是apache的一个开源项目iBatis, 2010年这个项目由apache software foundation迁移到了google code,并且改名为MyBatis 。2013年11月迁移到Github。    

 MyBatis是一个优秀的持久层框架,它对jdbc的操作数据库的过程进行封装,使开发者只需要关注 SQL 本身,而不需要花费精力去处理例如注册驱动、创建connection、创建statement、手动设置参数、结果集检索等jdbc繁杂的过程代码

Mybatis通过xml或注解的方式将要执行的各种statement(statement、preparedStatemnt、CallableStatement)配置起来,并通过java对象和statement中的sql进行映射生成最终执行的sql语句,最后由mybatis框架执行sql并将结果映射成java对象并返回。

2、使用jdbc编程问题总结

2.1   创建mysql数据库

先导入创建数据库的sql脚本导入到数据库中。

2.2   创建工程

开发环境:

eclipse mars

Jdk1.7

1、创建一个java工程。

2、导入jar包。此时需要mysql 的数据库驱动。

2.3   jdbc编程步骤:

1、  加载数据库驱动

2、  创建并获取数据库链接

3、  创建jdbc statement对象

4、  设置sql语句

5、  设置sql语句中的参数(使用preparedStatement)

6、  通过statement执行sql并获取结果

7、  sql执行结果进行解析处理

8、  释放资源(resultSetpreparedstatementconnection)

2.4   jdbc程序

public static void main(String[] args) {
          Connection connection = null;
          PreparedStatementprepared Statement = null;
          ResultSet resultSet = null;    
          try {
             //加载数据库驱动
             Class.forName('com.mysql.jdbc.Driver');
             //通过驱动管理类获取数据库链接
             connection=  DriverManager.getConnection('jdbc:mysql://localhost:3306/mybatis?characterEncoding=utf-8', 'root', 'root');
             //定义sql语句 ?表示占位符
          String sql = 'select * from user whereusername = ?';
             //获取预处理statement
             preparedStatement= connection.prepareStatement(sql);
             //设置参数,第一个参数为sql语句中参数的序号(从1开始),第二个参数为设置的参数值
             preparedStatement.setString(1,'王五');
             //向数据库发出sql执行查询,查询出结果集
             resultSet=  preparedStatement.executeQuery();
             //遍历查询结果集
             while(resultSet.next()){
System.out.println(resultSet.getString('id')+'  '+resultSet.getString('username'));
             }
          }catch (Exception e) {
             e.printStackTrace();
          }finally{
             //释放资源
             if(resultSet!=null){
                 try {
                    resultSet.close();
                 }catch (SQLException e) {
                    // TODO Auto-generated catch block
                     e.printStackTrace();
                 }
             }
             if(preparedStatement!=null){
                 try {
                    preparedStatement.close();
                 }catch (SQLException e) {
                    // TODO Auto-generated catch block
                    e.printStackTrace();
                 }
             }
             if(connection!=null){
                 try {
                    connection.close();
                 }catch (SQLException e) {
                    // TODO Auto-generated catch block
                    e.printStackTrace();
                 }
             }
          }
      }

上边使用jdbc的原始方法(未经封装)实现了查询数据库表记录的操作。

2.5   jdbc问题总结如下:

1、数据库链接创建、释放频繁造成系统资源浪费从而影响系统性能,如果使用数据库链接池可解决此问题。

2、Sql语句在代码中硬编码,造成代码不易维护,实际应用sql变化的可能较大,sql变动需要改变java代码。

3、使用preparedStatement向占有位符号传参数存在硬编码,因为sql语句的where条件不一定,可能多也可能少,修改sql还要修改代码,系统不易维护。

4、对结果集解析存在硬编码(查询列名),sql变化导致解析代码变化,系统不易维护,如果能将数据库记录封装成pojo对象解析比较方便。

3 、Mybatis架构

 

1、  mybatis配置

SqlMapConfig.xml,此文件作为mybatis的全局配置文件,配置了mybatis的运行环境等信息。

mapper.xml文件即sql映射文件,文件中配置了操作数据库的sql语句。此文件需要在SqlMapConfig.xml中加载。

2、  通过mybatis环境等配置信息构造SqlSessionFactory即会话工厂

3、  由会话工厂创建sqlSession即会话,操作数据库需要通过sqlSession进行。

4、  mybatis底层自定义了Executor执行器接口操作数据库,Executor接口有两个实现,一个是基本执行器、一个是缓存执行器。

5、  Mapped Statement也是mybatis一个底层封装对象,它包装了mybatis配置信息及sql映射信息等。mapper.xml文件中一个sql对应一个Mapped Statement对象,sql的id即是Mapped statement的id。

6、  Mapped Statement对sql执行输入参数进行定义,包括HashMap、基本类型、pojo,Executor通过MappedStatement在执行sql前将输入的java对象映射至sql中,输入参数映射就是jdbc编程中对preparedStatement设置参数。

7、  Mapped Statement对sql执行输出结果进行定义,包括HashMap、基本类型、pojo,Executor通过MappedStatement在执行sql后将输出结果映射至java对象中,输出结果映射过程相当于jdbc编程中对结果的解析处理过程。

4、Mybatis入门程序

4.1   mybatis下载

mybaits的代码由github.com管理,地址:https://github.com/mybatis/mybatis-3/releases

 

mybatis-3.2.7.jar----mybatis的核心包

lib----mybatis的依赖包

mybatis-3.2.7.pdf----mybatis使用手册


4.2   需求

实现以下功能:

根据用户id查询一个用户信息

根据用户名称模糊查询用户信息列表

添加用户

更新用户

删除用户

 

4.3   工程搭建

4.3.1  第一步:创建java工程

使用eclipse创建java工程,jdk使用1.7.0_72

4.3.2  第二步:加入jar

加入mybatis核心包、依赖包、数据驱动包。


 

4.3.3  第三步:log4j.properties

classpath下创建log4j.properties如下:

# Global logging configuration
log4j.rootLogger=DEBUG, stdout
# Console output...
log4j.appender.stdout=org.apache.log4j.ConsoleAppender
log4j.appender.stdout.layout=org.apache.log4j.PatternLayout
log4j.appender.stdout.layout.ConversionPattern=%5p [%t] - %m%n

mybatis默认使用log4j作为输出日志信息。

 

4.3.4  第四步:SqlMapConfig.xml

classpath下创建SqlMapConfig.xml,如下:

<>xml version='1.0' encoding='UTF-8'?>
PUBLIC '-////DTDConfig 3.0//EN'
'http:///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://localhost:3306/mybatis?characterEncoding=utf-8'/>
            property name='username'value='root' />
            property name='password'value='root' />
         dataSource>
     environment>
  environments>
configuration>

SqlMapConfig.xmlmybatis核心配置文件,上边文件的配置内容为数据源、事务管理。

4.3.5  第五步:po

Po类作为mybatis进行sql映射使用,po类通常与数据库表对应,User.java如下:

public class User {
   private int id;
   private String username;// 用户姓名
   private String sex;// 性别
   private Date birthday;// 生日
   private String address;// 地址
生成get/set……发法

4.3.6  第六步:sql映射文件

classpath下的sqlmap目录下创建sql映射文件Users.xml

<>xml version='1.0' encoding='UTF-8'?>
PUBLIC '-////DTD Mapper 3.0//EN'
'http:///dtd/mybatis-3-mapper.dtd'>

mapper namespace='test'>
mapper>

namespace 命名空间,用于隔离sql语句,后面会讲另一层非常重要的作用。


4.3.7  第七步:加载映射文件

mybatis框架需要加载映射文件,将Users.xml添加在SqlMapConfig.xml,如下:


      mapper resource='sqlmap/User.xml'/>
mappers>

4.4   根据id查询用户信息

4.4.1  映射文件:

user.xml中添加:


   select id='findUserById' parameterType='int' resultType='cn.itcast.mybatis.po.User'>
      select* from user where id = #{id}
   select>

parameterType:定义输入到sql中的映射类型,#{id}表示使用preparedstatement设置占位符号并将输入变量id传到sql

resultType定义结果映射类型。 

4.4.2  测试程序:

public class Mybatis_first {
   //会话工厂
   private SqlSessionFactory sqlSessionFactory;
   @Before
   publicvoid createSqlSessionFactory() throws IOException {
      // 配置文件
      Stringresource = 'SqlMapConfig.xml';
      InputStreaminputStream = Resources.getResourceAsStream(resource);
      // 使用SqlSessionFactoryBuilder从xml配置文件中创建SqlSessionFactory
      sqlSessionFactory = new SqlSessionFactoryBuilder()
             .build(inputStream);
   }
   // 根据 id查询用户信息
  @Test
  public void testFindUserById() {
     // 数据库会话实例
     SqlSessionsql Session = null;
     try {
         // 创建数据库会话实例sqlSession
         sqlSession= sqlSessionFactory.openSession();
         // 查询单个记录,根据用户id查询用户信息
         Useruser = sqlSession.selectOne('test.findUserById', 10);
         // 输出用户信息
         System.out.println(user);
     } catch (Exception e) {
         e.printStackTrace();
     } finally {
         if (sqlSession != null) {
            sqlSession.close();
         }
     }
  }
}

4.5   根据用户名查询用户信息

4.5.1  映射文件:

user.xml中添加:


   select id='findUserByUsername'parameterType='java.lang.String'
          resultType='cn.itcast.mybatis.po.User'>

      select * from user where usernamelike '%${value}%'
   select>

parameterType:定义输入到sql中的映射类型,${value}表示使用参数将${value}替换,做字符串的拼接。

注意:如果是取简单数量类型的参数,括号中的值必须为value

resultType定义结果映射类型。

4.5.2  测试程序:

// 根据用户名称模糊查询用户信息
  @Test
  public void testFindUserByUsername() {
     // 数据库会话实例
     SqlSessionsqlSession = null;
     try {
         // 创建数据库会话实例sqlSession
         sqlSession= sqlSessionFactory.openSession();
         // 查询单个记录,根据用户id查询用户信息
         Listlist = sqlSession.selectList('test.findUserByUsername', '张');
         System.out.println(list.size());
     } catch (Exception e) {
         e.printStackTrace();
     } finally {
         if (sqlSession != null) {
            sqlSession.close();
         }
     }
   }

4.6   小结

4.6.1  #{}${}

#{}表示一个占位符号,通过#{}可以实现preparedStatement向占位符中设置值,自动进行java类型和jdbc类型转换,#{}可以有效防止sql注入。 #{}可以接收简单类型值或pojo属性值。如果parameterType传输单个简单类型值,#{}括号中可以是value或其它名称。

 

${}表示拼接sql串,通过${}可以将parameterType 传入的内容拼接在sql中且不进行jdbc类型转换, ${}可以接收简单类型值或pojo属性值,如果parameterType传输单个简单类型值,${}括号中只能是value

 

4.6.2  parameterTyperesultType

parameterType:指定输入参数类型,mybatis通过ognl从输入对象中获取参数值拼接在sql中。

resultType:指定输出结果类型,mybatissql查询结果的一行记录数据映射为resultType指定类型的对象。

 

4.6.3  selectOneselectList

selectOne查询一条记录,如果使用selectOne查询多条记录则抛出异常:

org.apache.ibatis.exceptions.TooManyResultsException: Expected one result (or null) to bereturned by selectOne(), but found: 3

 atorg.apache.ibatis.session.defaults.DefaultSqlSession.selectOne(DefaultSqlSession.java:70)

 

selectList可以查询一条或多条记录。


4.7   添加用户

4.7.1  映射文件:

SqlMapConfig.xml中添加:


   insert id='insertUser'parameterType='cn.javaxxf.mybatis.po.User'>
     insert into user(username,birthday,sex,address)
     values(#{username},#{birthday},#{sex},#{address})
   insert>

 4.7.2  测试程序:

// 添加用户信息
   @Test
   public void testInsert() {
      // 数据库会话实例
      SqlSessionsqlSession = null;
      try {
          // 创建数据库会话实例sqlSession
          sqlSession= sqlSessionFactory.openSession();
          // 添加用户信息
          Useruser = newUser();
          user.setUsername('张小明');
          user.setAddress('河南郑州');
          user.setSex('1');
          user.setPrice(1999.9f);
          sqlSession.insert('test.insertUser', user);
          //提交事务
         sqlSession.commit();
      }catch (Exception e) {
          e.printStackTrace();
      }finally {
          if (sqlSession != null) {
             sqlSession.close();
          }
      }
   }

4.7.3  mysql自增主键返回

通过修改sql映射文件,可以将mysql自增主键返回:

<>'insertUser' parameterType='cn.javaxxf.mybatis.po.User'>
     
      selectKey keyProperty='id' order='AFTER' resultType='java.lang.Integer'>
          selectLAST_INSERT_ID()
      selectKey>
      insert into user(username,birthday,sex,address)
       values(#{username},#{birthday},#{sex},#{address});
   insert>

添加selectKey实现将主键返回

keyProperty:返回的主键存储在pojo中的哪个属性

order:selectKey的执行顺序,是相对与insert语句来说,由于mysql的自增原理执行完insert语句之后才将主键生成,所以这里selectKey的执行顺序为after

resultType:返回的主键是什么类型

LAST_INSERT_ID():mysql的函数,返回auto_increment自增列新记录id值。

 

4.7.4  Mysql使用 uuid实现主键

需要增加通过select uuid()得到uuid

insert  id='insertUser'parameterType='cn.itcast.mybatis.po.User'>
selectKey resultType='java.lang.String' order='BEFORE'
keyProperty='id'>

select uuid()
selectKey>
insertinto user(id,username,birthday,sex,address)
      values(#{id},#{username},#{birthday},#{sex},#{address})
insert>

注意这里使用的order是“BEFORE

4.8   删除用户

4.8.1  映射文件:


   delete id='deleteUserById'parameterType='int'>
      deletefrom user where id=#{id}
   delete>

4.8.2  测试程序:

// 根据id删除用户
   @Test
   public void testDelete() {
      // 数据库会话实例
      SqlSessionsql Session = null;
      try {
          // 创建数据库会话实例sqlSession
          sqlSession= sqlSessionFactory.openSession();
          // 删除用户
          sqlSession.delete('test.deleteUserById',18);
          // 提交事务
          sqlSession.commit();
      }catch (Exception e) {
          e.printStackTrace();
      }finally {
          if (sqlSession != null) {
             sqlSession.close();
          }
      }
   }

4.9   修改用户

4.9.1  映射文件


   update id='updateUser'parameterType='cn.javaxxf.mybatis.po.User'>
      updateuser set username=#{username},birthday=#{birthday},sex=#{sex},address=#{address}
      whereid=#{id}
   update>

4.9.2  测试程序

 

// 更新用户信息
   @Test
   public void testUpdate() {
      // 数据库会话实例
      SqlSessionsql Session = null;
      try {
          // 创建数据库会话实例sqlSession
          sqlSession= sqlSessionFactory.openSession();
          // 添加用户信息
          Useruser = newUser();
          user.setId(16);
          user.setUsername('张小明');
          user.setAddress('河南郑州');
          user.setSex('1');
          user.setPrice(1999.9f);
          sqlSession.update('test.updateUser', user);
          // 提交事务
          sqlSession.commit();
      }catch (Exception e) {
          e.printStackTrace();
      }finally {
          if (sqlSession != null) {
             sqlSession.close();
          }
      }
   }

4.10     Mybatis解决jdbc编程的问题

1、  数据库链接创建、释放频繁造成系统资源浪费从而影响系统性能,如果使用数据库链接池可解决此问题。

解决:SqlMapConfig.xml中配置数据链接池,使用连接池管理数据库链接。

2、  Sql语句写在代码中造成代码不易维护,实际应用sql变化的可能较大,sql变动需要改变java代码。

解决:Sql语句配置在XXXXmapper.xml文件中与java代码分离。

3、  sql语句传参数麻烦,因为sql语句的where条件不一定,可能多也可能少,占位符需要和参数一一对应。

解决:Mybatis自动将java对象映射至sql语句,通过statement中的parameterType定义输入参数的类型。

4、  对结果集解析麻烦,sql变化导致解析代码变化,且解析前需要遍历,如果能将数据库记录封装成pojo对象解析比较方便。

解决:Mybatis自动将sql执行结果映射至java对象,通过statement中的resultType定义输出结果的类型。

4.11     mybatishibernate不同

Mybatishibernate不同,它不完全是一个ORM框架,因为MyBatis需要程序员自己编写Sql语句,不过mybatis可以通过XML或注解方式灵活配置要运行的sql语句,并将java对象和sql语句映射生成最终执行的sql,最后将sql执行的结果再映射生成java对象。

 

Mybatis学习门槛低,简单易学,程序员直接编写原生态sql,可严格控制sql执行性能,灵活度高,非常适合对关系数据模型要求不高的软件开发,例如互联网软件、企业运营类软件等,因为这类软件需求变化频繁,一但需求变化要求成果输出迅速。但是灵活的前提是mybatis无法做到数据库无关性,如果需要实现支持多种数据库的软件则需要自定义多套sql映射文件,工作量大。

 

Hibernate对象/关系映射能力强,数据库无关性好,对于关系模型要求高的软件(例如需求固定的定制化软件)如果用hibernate开发可以节省很多代码,提高效率。但是Hibernate的学习门槛高,要精通门槛更高,而且怎么设计O/R映射,在性能和对象模型之间如何权衡,以及怎样用好Hibernate需要具有很强的经验和能力才行。

总之,按照用户的需求在有限的资源环境下只要能做出维护性、扩展性良好的软件架构都是好架构,所以框架只有适合才是最好。 

    本站是提供个人知识管理的网络存储空间,所有内容均由用户发布,不代表本站观点。请注意甄别内容中的联系方式、诱导购买等信息,谨防诈骗。如发现有害或侵权内容,请点击一键举报。
    转藏 分享 献花(0

    0条评论

    发表

    请遵守用户 评论公约

    类似文章 更多