MyBatis—学习笔记#day02

MyBatis

Posted by DiCaprio on July 12, 2019

目录

1、回顾mybatis自定义和环境搭建+完善自定义Mybatis的注解开发

2、Mybatis基于代理Dao的CRUD操作(重点内容)

2.1映射配置文件中配置

2.2模糊查询的两种配置 

2.3#{} 与${}的区别

 2.4获取自增id

2.5Mybatis 与 JDBC编程的比较

3、CRUD中可能遇到的问题:参数的传递以及返回值的封装

3.1parameterType  配置参数

使用说明

注意事项

OGNL表达式:

3.2 resultType  配置结果类型

3.3 resultMap结果类型

3.3.1定义

4、介绍Mybatis基于传统dao方式的使用(自己编写dao的实现类)    了解

4.1持久层 Dao  实现类

4.2 持久层映射配置

Dao实现类执行过程分析—查询方法

5、主配置文件 SqlMapConfig.xml

5.1SqlMapConfig.xml  中配置的内容和顺序

5.2properties

第一种

第二种

5.3typeAliases (类型别名)

自定义别名:

5.4mappers映射器


1、回顾mybatis自定义和环境搭建+完善自定义Mybatis的注解开发

自定义mybatis开发流程图

MyBatis—学习笔记#day01源码:https://github.com/DiCaprio17/MyBatis

2、Mybatis基于代理Dao的CRUD操作(重点内容)

2.1映射配置文件中配置

细节:
resultType属性:用于指定结果集的类型。代表参数的类型,因为我们要传入的是一个类的对象,所以类型就写类的全名称。
parameterType属性:用于指定传入参数的类型。
sql语句中使用#{} 字符:它代表占位符,相当于原来 jdbc 部分所学的,都是用于执行语句时替换实际的数据。具体的数据是由#{}里面的内容决定的。
#{} 中内容的写法:数据类型是基本类型,若只有一个时,此处可以随意写,否则填对象的属性名称,它用的是 ognl 表达式。

#{user.username}它会先去找 user 对象,然后在 user 对象中找到 username 属性,并调用
getUsername()方法把值取出来。但是我们在 parameterType 属性上指定了实体类名称,所以可以省略 user.
而直接写 username。

2.2模糊查询的两种配置 

我们在配置文件中没有加入%来作为模糊查询的条件,所以在传入字符串实参时,就需要给定模糊查询的标
识%。配置文件中的#{username}也只是一个占位符,所以 SQL 语句显示为“?”

可以发现,我们在程序代码中就不需要加入模糊查询的匹配符%了,这两种方式的实现效果是一样的,但执行
的语句是不一样的

2.3#{} 与${}的区别

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

 2.4获取自增id

  1. <!-- 保存用户 -->
  2. <insert id="saveUser" parameterType="user">
  3. <!-- 配置插入操作后,获取插入数据的id -->
  4. <!-- keyProperty:属性id,keyColumn:数据库id,resultType:数据类型,order:(保存)前做后做? -->
  5. <selectKey keyProperty="userId" keyColumn="id" resultType="int" order="AFTER">
  6. select last_insert_id();
  7. </selectKey>
  8. insert into user(username,address,sex,birthday)values(#{username},#{address},#{sex},#{birthday});
  9. </insert>

2.5Mybatis 与 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 定义输出结果的类型。

3、CRUD中可能遇到的问题:参数的传递以及返回值的封装

3.1parameterType  配置参数

使用说明

该属性的取值可以
是基本类型,引用类型(例如:String 类型),还可以是实体类类型(POJO 类)。同时也可以使用实体类的包装

注意事项

基本类型和 String 我 们可以直接写类型名称 ,也可以使用包名.类名的方式,例如 :java.lang.String。
实体类类型,目前我们只能使用全限定类名。

究其原因,是 mybaits 在加载时已经把常用的数据类型注册了别名,从而我们在使用时可以不写包名,而我们的是实体类并没有注册别名,所以必须写全限定类名。

OGNL表达式:

       Object Graphic Navigation Language

       对象       图    导航          语言

       它是通过对象的取值方法来获取数据。在写法上把get给省略了。

       比如:我们获取用户的名称

              类中的写法:user.getUsername();

              OGNL表达式写法:user.username

       mybatis中为什么能直接写username,而不用user.呢:

              因为在parameterType中已经提供了属性所属的类,所以此时不需要写对象名

3.2 resultType  配置结果类型

resultType 属性可以指定结果集的类型,它支持基本类型和实体类类型。
我们在前面的 CRUD 案例中已经对此属性进行过应用了。
需要注意的是,它和 parameterType 一样,如果注册过类型别名的,可以直接使用别名。没有注册过的必须使用全限定类名。例如:我们的实体类此时必须是全限定类名(今天最后一个章节会讲解如何配置实体类的别名)
同时,当是实体类名称是,还有一个要求,实体类中的属性名称必须和查询语句中的列名保持一致,否则无法实现封装。

但在Windows下的MySQL数据字段名不区分大小写,所以username和userName相同,因此可以封装,其他则不行

解决方法一:给属性起别名

但当属性很多时,起别名则显得麻烦,因此

解决方法二:3.3 resultMap结果类型,配置 查询结果的列名和实体类的属性名的对应关系

3.3 resultMap结果类型

resultMap 标签可以建立查询的列名和实体类的属性名称不一致时建立对应关系。从而实现封装。
在 select 标签中使用 resultMap 属性指定引用即可。同时 resultMap 可以实现将查询结果映射为复杂类型的 pojo,比如在查询结果映射对象中包括 pojo 和 list 实现一对一查询和一对多查询。

3.3.1定义

type 属性:指定实体类的全限定类名
id 属性:给定一个唯一标识,是给查询 select 标签引用用的

id 标签:用于指定主键字段
result 标签:用于指定非主键字段
column 属性:用于指定数据库列名
property 属性:用于指定实体类属性名称

  1. <resultMap type="com.mybatis.domain.User" id="userMap">
  2. <id column="id" property="userId"/>
  3. <result column="username" property="userName"/>
  4. <result column="sex" property="userSex"/>
  5. <result column="address" property="userAddress"/>
  6. <result column="birthday" property="userBirthday"/>
  7. </resultMap>

 

4、介绍Mybatis基于传统dao方式的使用(自己编写dao的实现类)    了解

4.1持久层 Dao  实现类

  1. package com.mybatis.dao.impl;
  2. import com.itheima.dao.IUserDao;
  3. import com.itheima.domain.User;
  4. import org.apache.ibatis.session.SqlSession;
  5. import org.apache.ibatis.session.SqlSessionFactory;
  6. import java.util.List;
  7. public class UserDaoImpl implements IUserDao {
  8. private SqlSessionFactory factory;
  9. public UserDaoImpl(SqlSessionFactory factory){
  10. this.factory = factory;
  11. }
  12. @Override
  13. public List<User> findAll() {
  14. //1.根据factory获取SqlSession对象
  15. SqlSession session = factory.openSession();
  16. //2.调用SqlSession中的方法,实现查询列表
  17. List<User> users = session.selectList("com.itheima.dao.IUserDao.findAll");//参数就是能获取配置信息的key
  18. //3.释放资源
  19. session.close();
  20. return users;
  21. }
  22. @Override
  23. public void saveUser(User user) {
  24. //1.根据factory获取SqlSession对象
  25. SqlSession session = factory.openSession();
  26. //2.调用方法实现保存
  27. session.insert("com.itheima.dao.IUserDao.saveUser",user);
  28. //3.提交事务
  29. session.commit();
  30. //4.释放资源
  31. session.close();
  32. }
  33. @Override
  34. public void updateUser(User user) {
  35. //1.根据factory获取SqlSession对象
  36. SqlSession session = factory.openSession();
  37. //2.调用方法实现更新
  38. session.update("com.itheima.dao.IUserDao.updateUser",user);
  39. //3.提交事务
  40. session.commit();
  41. //4.释放资源
  42. session.close();
  43. }
  44. @Override
  45. public void deleteUser(Integer userId) {
  46. //1.根据factory获取SqlSession对象
  47. SqlSession session = factory.openSession();
  48. //2.调用方法实现更新
  49. session.update("com.itheima.dao.IUserDao.deleteUser",userId);
  50. //3.提交事务
  51. session.commit();
  52. //4.释放资源
  53. session.close();
  54. }
  55. @Override
  56. public User findById(Integer userId) {
  57. //1.根据factory获取SqlSession对象
  58. SqlSession session = factory.openSession();
  59. //2.调用SqlSession中的方法,实现查询一个
  60. User user = session.selectOne("com.itheima.dao.IUserDao.findById",userId);
  61. //3.释放资源
  62. session.close();
  63. return user;
  64. }
  65. @Override
  66. public List<User> findByName(String username) {
  67. //1.根据factory获取SqlSession对象
  68. SqlSession session = factory.openSession();
  69. //2.调用SqlSession中的方法,实现查询列表
  70. List<User> users = session.selectList("com.itheima.dao.IUserDao.findByName",username);
  71. //3.释放资源
  72. session.close();
  73. return users;
  74. }
  75. @Override
  76. public int findTotal() {
  77. //1.根据factory获取SqlSession对象
  78. SqlSession session = factory.openSession();
  79. //2.调用SqlSession中的方法,实现查询一个
  80. Integer count = session.selectOne("com.itheima.dao.IUserDao.findTotal");
  81. //3.释放资源
  82. session.close();
  83. return count;
  84. }
  85. }

4.2 持久层映射配置

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <!DOCTYPE mapper
  3. PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
  4. "http://mybatis.org/dtd/mybatis-3-mapper.dtd">
  5. <mapper namespace="com.mybatis.dao.IUserDao">
  6. <!-- 查询所有 -->
  7. <select id="findAll" resultType="com.mybatis.domain.User">
  8. select * from user;
  9. </select>
  10. <!-- 保存用户 -->
  11. <insert id="saveUser" parameterType="com.mybatis.domain.User">
  12. <!-- 配置插入操作后,获取插入数据的id -->
  13. <selectKey keyProperty="id" keyColumn="id" resultType="int" order="AFTER">
  14. select last_insert_id();
  15. </selectKey>
  16. insert into user(username,address,sex,birthday)values(#{username},#{address},#{sex},#{birthday});
  17. </insert>
  18. <!-- 更新用户 -->
  19. <update id="updateUser" parameterType="com.mybatis.domain.User">
  20. update user set username=#{username},address=#{address},sex=#{sex},birthday=#{birthday} where id=#{id}
  21. </update>
  22. <!-- 删除用户-->
  23. <delete id="deleteUser" parameterType="java.lang.Integer">
  24. delete from user where id = #{uid}
  25. </delete>
  26. <!-- 根据id查询用户 -->
  27. <select id="findById" parameterType="INT" resultType="com.mybatis.domain.User">
  28. select * from user where id = #{uid}
  29. </select>
  30. <!-- 根据名称模糊查询 -->
  31. <select id="findByName" parameterType="string" resultType="com.mybatis.domain.User">
  32. select * from user where username like #{name}
  33. </select>
  34. <!-- 获取用户的总记录条数 -->
  35. <select id="findTotal" resultType="int">
  36. select count(id) from user;
  37. </select>
  38. </mapper>

Dao实现类执行过程分析—查询方法

5、主配置文件 SqlMapConfig.xml

5.1SqlMapConfig.xml  中配置的内容和顺序

-properties (属性)
    --property
-settings(全局配置参数)
   --setting
-typeAliases (类型别名)
   --typeAliase
   --package
-typeHandlers(类型处理器)
-objectFactory(对象工厂)
-plugins(插件)
-environments(环境集合属性对象)
   --environment(环境子属性对象)
      ---transactionManager(事务管理)
      ---dataSource(数据源)
-mappers (映射器)
   --mapper
   --package

5.2properties

第一种

  1. <properties>
  2. <property name="jdbc.driver" value="com.mysql.jdbc.Driver"/>
  3. <property name="jdbc.url" value="jdbc:mysql://localhost:3306/eesy"/>
  4. <property name="jdbc.username" value="root"/>
  5. <property name="jdbc.password" value="1234"/>
  6. </properties>

第二种

在 classpath  下定义 db.properties  文件

  1. jdbc.driver=com.mysql.jdbc.Driver
  2. jdbc.url=jdbc:mysql://localhost:3306/eesy
  3. jdbc.username=root
  4. jdbc.password=1234

properties 配置

  1. <!-- 配置连接数据库的信息
  2. resource 属性:用于指定 properties 配置文件的位置,要求配置文件必须在类路径下
  3. resource="jdbcConfig.properties"
  4. url 属性:
  5. URL: Uniform Resource Locator 统一资源定位符
  6. http://localhost:8080/mystroe/CategoryServlet URL
  7. 协议 主机 端口 URI
  8. URI:Uniform Resource Identifier 统一资源标识符
  9. /mystroe/CategoryServlet
  10. 它是可以在 web 应用中唯一定位一个资源的路径
  11. -->
  12. <properties url=file:///D:/IdeaProjects/mybatisCRUD/src/main/resources/jdbcConfig.properties">
  13. </properties>

或者

dataSource引用配置

  1. <dataSource type="POOLED">
  2. <property name="driver" value="${jdbc.driver}"/>
  3. <property name="url" value="${jdbc.url}"/>
  4. <property name="username" value="${jdbc.username}"/>
  5. <property name="password" value="${jdbc.password}"/>
  6. </dataSource>

5.3typeAliases (类型别名)

自定义别名:

  1. 在 SqlMapConfig.xml 中配置:
  2. <typeAliases>
  3. <!-- 单个别名定义 -->
  4. <typeAlias alias="user" type="com.mybatis.domain.User"/>
  5. <!-- 批量别名定义,扫描整个包下的类,别名为类名(首字母大写或小写都可以) -->
  6. <package name="com.mybatis.domain"/>
  7. <package name=" 其它包 "/>
  8. </typeAliases>

typeAlias用于配置别名,type指定实体类全限定类名,alias指定别名,当指定了别名就不再区分其大小写 

  

5.4mappers映射器

<mapper resource=" " />

使用相对于类路径的资源
如:<mapper resource="com/itheima/dao/IUserDao.xml" />

<mapper class=" " />

使用 mapper 接口类路径
如:<mapper class="com.mybatis.dao.UserDao"/>
注意:此种方法要求 mapper  接口名称和 mapper 映射文件名称相同,且放在同一个目录中

<package name=""/>

注册指定包下的所有 mapper 接口
如:<package name="cn.itcast.mybatis.mapper"/>
注意:此种方法要求  mapper  接口名称和 mapper  映射文件名称相同,且放在同一个目录中

源码:https://github.com/DiCaprio17/MyBatis/tree/master/mybatis_day02 


Related Issues not found

Please contact @dicaprio17 to initialize the comment