对Mapper 中几种update的区别说明

这两个update都是使用generator生成的mapper.xml文件中,对dao层的更新操作

update

更新传回数据的所有字段,没有传回的字段保持原样。

updateByPrimaryKey

对实体类的字段全部更新(不判断是否为Null),即如果字段为空就更新为空;

updateByPrimaryKeySelective

会对实体类字段进行判断再更新(如果为Null就忽略更新),如果字段为空,忽略不更新;

补充知识:mapper中insert、update、delete、select、resultMap的用法

这里介绍mapper映射文件的配置, 这是mybatis的核心之一,一定要学好。

在mapper文件中,以mapper作为根节点,其下面可以配置的元素节点有: select, insert, update, delete, cache, cache-ref, resultMap, sql 。

先来看看 insert, update, delete 怎么配置, 能配置哪些元素吧:

<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
   PUBLIC "-//ibatis.apache.org//DTD Mapper 3.0//EN"
   "http://ibatis.apache.org/dtd/ibatis-3-mapper.dtd"> 

<!-- mapper 为根元素节点, 一个namespace对应一个dao -->
<mapper namespace="com.dy.dao.UserDao">
  <insert
   <!-- 1. id (必须配置)
    id是命名空间中的唯一标识符,可被用来代表这条语句。 一个命名空间(namespace) 对应一个dao接口,
    这个id也应该对应dao里面的某个方法(相当于方法的实现),因此id 应该与方法名一致 -->

   id="insertUser"

   <!-- 2. parameterType (可选配置, 默认为mybatis自动选择处理)
    将要传入语句的参数完全限定类名或别名,如果不配置,mybatis会通过ParameterHandler 根据参数类型默认选择合适的typeHandler进行处理
    parameterType 主要指定参数类型,可以是int, short, long, string等类型,也可以是复杂类型(如对象) -->

   parameterType="com.demo.User"

   <!-- 3. flushCache (可选配置,默认配置为true)
    将其设置为 true,任何时候只要语句被调用,都会导致本地缓存和二级缓存都会被清空,
    默认值:true(对应插入、更新和删除语句) -->

   flushCache="true"

   <!-- 4. statementType (可选配置,默认配置为PREPARED)
    STATEMENT,PREPARED 或 CALLABLE 的一个。这会让 MyBatis 分别使用 Statement,PreparedStatement 或 CallableStatement,默认值:PREPARED。 -->

   statementType="PREPARED"

   <!-- 5. keyProperty (可选配置, 默认为unset)
    (仅对 insert 和 update 有用)唯一标记一个属性,MyBatis 会通过 getGeneratedKeys 的返回值或者通过 insert 语句的 selectKey 子元素设置它的键值,默认:unset。如果希望得到多个生成的列,也可以是逗号分隔的属性名称列表。 -->

   keyProperty=""

   <!-- 6. keyColumn   (可选配置)
    (仅对 insert 和 update 有用)通过生成的键值设置表中的列名,这个设置仅在某些数据库(像 PostgreSQL)是必须的,当主键列不是表中的第一列的时候需要设置。如果希望得到多个生成的列,也可以是逗号分隔的属性名称列表。 -->

   keyColumn=""

   <!-- 7. useGeneratedKeys (可选配置, 默认为false)
    (仅对 insert 和 update 有用)这会令 MyBatis 使用 JDBC 的 getGeneratedKeys 方法来取出由数据库内部生成的主键(比如:像 MySQL 和 SQL Server 这样的关系数据库管理系统的自动递增字段),默认值:false。 -->

   useGeneratedKeys="false"

   <!-- 8. timeout (可选配置, 默认为unset, 依赖驱动)
    这个设置是在抛出异常之前,驱动程序等待数据库返回请求结果的秒数。默认值为 unset(依赖驱动)。 -->

   timeout="20">

  <update
   id="updateUser"
   parameterType="com.demo.User"
   flushCache="true"
   statementType="PREPARED"
   timeout="20">

  <delete
   id="deleteUser"
   parameterType="com.demo.User"
   flushCache="true"
   statementType="PREPARED"
   timeout="20">
</mapper>

以上就是一个模板配置, 哪些是必要配置,哪些是根据自己实际需求,看一眼就知道了。

仔细观察上面parameterType, "com.dy.entity.User",包名要是再长点呢?那么这个地方,用上typeAliases别名,岂不是能跟长长的包名说拜拜了。在哪儿配? 当然是在mybatis 的全局配置文件(我这儿名字是mybatis-conf.xml),不要认为是在mapper的配置文件里面配置哈。

mybatis-conf.xml:

<typeAliases>
   <!--
   通过package, 可以直接指定package的名字, mybatis会自动扫描你指定包下面的javabean,
   并且默认设置一个别名,默认的名字为: javabean 的首字母小写的非限定类名来作为它的别名。
   也可在javabean 加上注解@Alias 来自定义别名, 例如: @Alias(user)
   <package name="com.dy.entity"/>
    -->
   <typeAlias alias="user" type="com.dy.entity.User"/>
 </typeAliases>

这样,一个别名就取好了,可以把上面的 com.dy.entity.User 都直接改为user 了。

我这儿数据库用的是mysql, 我把user表的主键id 设置了自动增长, 以上代码运行正常, 那么问题来了(当然,我不是要问学挖掘机哪家强),我要是换成oracle数据库怎么办? oracle 可是不支持id自增长啊? 怎么办?请看下面:

<!-- 对应userDao中的insertUser方法, -->
  <insert id="insertUser" parameterType="com.dy.entity.User">
      <!-- oracle等不支持id自增长的,可根据其id生成策略,先获取id -->

    <selectKey resultType="int" order="BEFORE" keyProperty="id">
       select seq_user_id.nextval as id from dual
    </selectKey>

    insert into user(id, name, password, age, deleteFlag)
      values(#{id}, #{name}, #{password}, #{age}, #{deleteFlag})
  </insert>

同理,如果我们在使用mysql的时候,想在数据插入后返回插入的id, 我们也可以使用 selectKey 这个元素:

<!-- 对应userDao中的insertUser方法, -->
  <insert id="insertUser" parameterType="com.dy.entity.User">
    <!-- mysql插入数据后,获取id -->
    <selectKey keyProperty="id" resultType="int" order="AFTER" >
        SELECT LAST_INSERT_ID() as id
    </selectKey>

    insert into user(id, name, password, age, deleteFlag)
        values(#{id}, #{name}, #{password}, #{age}, #{deleteFlag})
  </insert>

这儿,我们就简单提一下 <selectKey> 这个元素节点吧:

<selectKey
    <!-- selectKey 语句结果应该被设置的目标属性。如果希望得到多个生成的列,也可以是逗号分隔的属性名称列表。 -->
    keyProperty="id"
    <!-- 结果的类型。MyBatis 通常可以推算出来,但是为了更加确定写上也不会有什么问题。MyBatis 允许任何简单类型用作主键的类型,包括字符串。如果希望作用于多个生成的列,则可以使用一个包含期望属性的 Object 或一个 Map。 -->
    resultType="int"
    <!-- 这可以被设置为 BEFORE 或 AFTER。如果设置为 BEFORE,那么它会首先选择主键,设置 keyProperty 然后执行插入语句。如果设置为 AFTER,那么先执行插入语句,然后是 selectKey 元素 - 这和像 Oracle 的数据库相似,在插入语句内部可能有嵌入索引调用。 -->
    order="BEFORE"
    <!-- 与前面相同,MyBatis 支持 STATEMENT,PREPARED 和 CALLABLE 语句的映射类型,分别代表 PreparedStatement 和 CallableStatement 类型。 -->
    statementType="PREPARED">

上面介绍了insert、update、delete的用法,下面介绍select、resultMap的用法。select无疑是我们最常用,也是 最复杂的,mybatis通过resultMap能帮助我们很好地进行高级映射。下面就开始看看select 以及 resultMap的用法:

先看select的配置吧:

<select
    <!-- 1. id (必须配置)
    id是命名空间中的唯一标识符,可被用来代表这条语句。
    一个命名空间(namespace) 对应一个dao接口,
    这个id也应该对应dao里面的某个方法(相当于方法的实现),因此id 应该与方法名一致 -->

   id="selectPerson"

   <!-- 2. parameterType (可选配置, 默认为mybatis自动选择处理)
    将要传入语句的参数的完全限定类名或别名, 如果不配置,mybatis会通过ParameterHandler 根据参数类型默认选择合适的typeHandler进行处理
    parameterType 主要指定参数类型,可以是int, short, long, string等类型,也可以是复杂类型(如对象) -->
   parameterType="int"

   <!-- 3. resultType (resultType 与 resultMap 二选一配置)
     resultType用以指定返回类型,指定的类型可以是基本类型,可以是java容器,也可以是javabean -->
   resultType="hashmap"

   <!-- 4. resultMap (resultType 与 resultMap 二选一配置)
     resultMap用于引用我们通过 resultMap标签定义的映射类型,这也是mybatis组件高级复杂映射的关键 -->
   resultMap="personResultMap"

   <!-- 5. flushCache (可选配置)
     将其设置为 true,任何时候只要语句被调用,都会导致本地缓存和二级缓存都会被清空,默认值:false -->
   flushCache="false"

   <!-- 6. useCache (可选配置)
     将其设置为 true,将会导致本条语句的结果被二级缓存,默认值:对 select 元素为 true -->
   useCache="true"

   <!-- 7. timeout (可选配置)
     这个设置是在抛出异常之前,驱动程序等待数据库返回请求结果的秒数。默认值为 unset(依赖驱动)-->
   timeout="10000"

   <!-- 8. fetchSize (可选配置)
     这是尝试影响驱动程序每次批量返回的结果行数和这个设置值相等。默认值为 unset(依赖驱动)-->
   fetchSize="256"

   <!-- 9. statementType (可选配置)
     STATEMENT,PREPARED 或 CALLABLE 的一个。这会让 MyBatis 分别使用 Statement,PreparedStatement 或 CallableStatement,默认值:PREPARED-->
   statementType="PREPARED"

   <!-- 10. resultSetType (可选配置)
     FORWARD_ONLY,SCROLL_SENSITIVE 或 SCROLL_INSENSITIVE 中的一个,默认值为 unset (依赖驱动)-->
   resultSetType="FORWARD_ONLY">

配置看起来总是这么多,不过实际常用的配置也就那么几个, 根据自己的需要吧,上面都已注明是否必须配置。

下面是针对select 的练手

其中,1个student可选择多个course进行学习。

<mapper namespace="com.dy.dao.CourseDao">

  <!--
    1.此处直接将resultType 设置为course, 一看就知道我设置了别名吧,如果没有设置别名,那么resultType = com.dy.entity.Course。
     2.可能细心的你会发现:Course.java中的属性名与数据库字段名不一致,下面,我就在sql语句中用了as, 使之匹配,当然方法不止一种,
       在学习了resultMap之后,你能看到一种更直观优雅的方式去将javabean中的属性与数据库字段名保持一致
     3.findCourseById 与CourseDao中findCourseById方法对应, 那么传入的参数名称以及类型也应该保持对应关系。
     4.可以看到,在sql语句中,通过#{}表达式可以获取参数。
     5.下面这条sql语句,实际上的形式是怎么样的?还记得之前说过,mybatis默认为preparedStatement吧,那么,用我们jdbc代码来看,它其实就是:
       select course_id as id, course_name as name, course_delete_flg as deleteFlag from t_course where course_id=?
   -->
  <select id="findCourseById" resultType="course" >
    select course_id as id, course_name as name, course_delete_flg as deleteFlag from t_course where course_id=#{courseId}
  </select>
</mapper>

上面我们针对course, 简单演示了 select的用法, 不过有个问题值得思考: 一个student可以对应多个course, 那么,在mybatis中如何处理这种一对多, 甚至于多对多,一对一的关系呢?

这儿,就不得不提到 resultMap 这个东西, mybatis的resultMap功能可谓十分强大,能够处理复杂的关系映射, 那么resultMap 该怎么配置呢? 别急,这就来了:

resultMap的配置:

<!--
    1.type 对应类型,可以是javabean, 也可以是其它
    2.id 必须唯一, 用于标示这个resultMap的唯一性,在使用resultMap的时候,就是通过id指定
   -->
  <resultMap type="" id="">

    <!-- id, 唯一性,注意啦,这个id用于标示这个javabean对象的唯一性, 不一定会是数据库的主键(不要把它理解为数据库对应表的主键)
      property属性对应javabean的属性名,column对应数据库表的列名
      (这样,当javabean的属性与数据库对应表的列名不一致的时候,就能通过指定这个保持正常映射了)
    -->
    <id property="" column=""/>

    <!-- result与id相比, 对应普通属性 -->
    <result property="" column=""/>

    <!-- constructor对应javabean中的构造方法-->
    <constructor>
      <!-- idArg 对应构造方法中的id参数 -->
      <idArg column=""/>
      <!-- arg 对应构造方法中的普通参数 -->
      <arg column=""/>
    </constructor>

    <!--
      collection,对应javabean中容器类型, 是实现一对多的关键
      property 为javabean中容器对应字段名
      column 为体现在数据库中列名
      ofType 就是指定javabean中容器指定的类型
    -->
    <collection property="" column="" ofType=""></collection>

    <!--
      association 为关联关系,是实现N对一的关键。
      property 为javabean中容器对应字段名
      column 为体现在数据库中列名
      javaType 指定关联的类型
     -->
    <association property="" column="" javaType=""></association>
  </resultMap>

以上这篇对Mapper 中几种update的区别说明就是小编分享给大家的全部内容了,希望能给大家一个参考,也希望大家多多支持我们。

(0)

相关推荐

  • 详解mybatis 批量更新数据两种方法效率对比

    上节探讨了批量新增数据,这节探讨批量更新数据两种写法的效率问题. 实现方式有两种, 一种用for循环通过循环传过来的参数集合,循环出N条sql, 另一种 用mysql的case when 条件判断变相的进行批量更新 下面进行实现. 注意第一种方法要想成功,需要在db链接url后面带一个参数  &allowMultiQueries=true 即:  jdbc:mysql://localhost:3306/mysqlTest?characterEncoding=utf-8&allowMulti

  • MyBatis注解方式之@Update/@Delete使用详解

    @Update 1. RoleMapper接口增加接口方法 /** * * * @Title: updateSysRoleById * * @Description: updateSysRoleById * * @param sysRole * @return * * @return: int */ @Update({ "update sys_role set role_name = #{roleName},enabled = #{enabled},create_by = #{createBy}

  • MybatisPlus中@TableField注解的使用详解

    实现 官方文档说明: com.baomidou.mybatisplus.annotations.TableField TableField注解新增属性 update 预处理 set 字段自定义注入 (讲解:比如我们使用mybatisplus自带的insert()方法向数据库插入数据时,假设我们给age字段赋值为1,但是我们在age字段上的@TableField注解里面加了update="%s+1",那么真真插入到数据库的值就是age=2,而不是age+1了) 例如: @TableFie

  • 解决Mybatis中mapper.xml文件update,delete及insert返回值问题

    最近写了几个非常简单的接口(CRUD),在单元测试的时候却出了问题,报错如下: Caused by: org.springframework.beans.factory.UnsatisfiedDependencyException: Error creating bean with name 'messageListener': Unsatisfied dependency expressed through field 'reviewCheckInfoService'; nested exce

  • 对Mapper 中几种update的区别说明

    这两个update都是使用generator生成的mapper.xml文件中,对dao层的更新操作 update 更新传回数据的所有字段,没有传回的字段保持原样. updateByPrimaryKey 对实体类的字段全部更新(不判断是否为Null),即如果字段为空就更新为空: updateByPrimaryKeySelective 会对实体类字段进行判断再更新(如果为Null就忽略更新),如果字段为空,忽略不更新: 补充知识:mapper中insert.update.delete.select.

  • 关于AngularJS中几种Providers的区别总结

    原文:https://xebia.com/blog/differ... 什么是Provider? AngularJS文档对provider的定义: provider是一个带有$get()方法的对象.injector调用$get方法创建一个新的service的实例.provider还有一些其他的方法,可以用来配置provider. AngularJS使用$provide注册新的providers.providers基本上都会创建一个新实例, 但每个provider只创建一次.$provide提供了

  • 分享下php5类中三种数据类型的区别

    public: 公有类型 在子类中可以通过self::var 来调用 public类型的方法或属性 可以通过parent::method 来调用父类中的方法 在实例中可以能过$obj->var 来调用 public类型的方法或属性 protected: 受保护类型 在子类中可以通过self::var 来调用 protected类型的方法或属性 可以通过parent::method 来调用父类中的方法 在实例中不能通过$obj->var 来调用 protected类型的方法或属性 private

  • C++中4种强制类型转换的区别详析

    前言 C++即支持C风格的类型转换,又有自己风格的类型转换.C风格的转换格式很简单,但是有不少缺点的: 1.转换太过随意,可以在任意类型之间转换.你可以把一个指向const对象的指针转换成指向非const对象的指针,把一个指向基类对象的指针转换成一个派生类对象的指针,这些转换之间的差距是非常巨大的,但是传统的C语言风格的类型转换没有区分这些. 2.C风格的转换没有统一的关键字和标示符.对于大型系统,做代码排查时容易遗漏和忽略. C++风格完美的解决了上面两个问题.1.对类型转换做了细分,提供了四

  • Java中生成随机数的4种方式与区别详解

    目录 在 Java 中,生成随机数的场景有很多,所以本文我们就来盘点一下 4 种生成随机数的方式,以及它们之间的区别和每种生成方式所对应的场景. 1.Random Random 类诞生于 JDK 1.0,它产生的随机数是伪随机数,也就是有规则的随机数.Random 使用的随机算法为 linear congruential pseudorandom number generator (LGC) 线性同余法伪随机数.在随机数生成时,随机算法的起源数字称为种子数(seed),在种子数的基础上进行一定的

  • MyBatis中多条件查询商品的三种方法及区别

    目录 一.Sql语句设置多个参数有几种方式 二.代码附上 一.Sql语句设置多个参数有几种方式 1.散装参数:需要使用@Param标记Sql语句中占位符处的名称例如 #{name} 2.实体类封装参数 只需要保证Sql中的参数名和实体类属性名对应上,即可设置成功BrandMapper.xml中的SQL语句不用动,把TestBrandMapper中的代码修改即可 3.Map集合 只需要保证Sql中的参数名和Map集合的键的名称对应上,即可设置成功.BrandMapper.xml中Sql语句不用改

  • Linux Shell中三种引号的用法及区别

    Linux Shell中有三种引号,分别为双引号(" ").单引号(' ')以及反引号(` `). 其中双引号对字符串中出现的$.''.`和\进行替换:单引号不进行替换,将字符串中所有字符作为普通字符输出,而反引号中字符串作为shell命令执行,并返回执行结果.具体含义如下: 双引号(" "):在双引号中,除了$, '', `和\以外所有的字符都解释成字符本身. 单引号(' '):在单引号中所有的字符包括特殊字符($,'',`和\)都将解释成字符本身而成为普通字符.

  • 浅谈java中math类中三种取整函数的区别

    math类中三大取整函数 1.ceil 2.floor 3.round 其实三种取整函数挺简单的.只要记住三个函数名翻译过来的汉语便能轻松理解三大函数,下面一一介绍 1.ceil,意思是天花板,java中叫做向上取整,大于等于该数字的最接近的整数 例: math.ceil(13.2)=14 math.ceil(-13.2)=-13 2.floor,意思是地板,java中叫做向下取整,小于等于该数字的最接近的整数 例: math.floor(13.2)=13 math.floor(-13.2)=-

  • 解析C++中四种强制类型转换的区别详解

    C++的四种强制类型转换,所以C++不是类型安全的.分别为:static_cast , dynamic_cast , const_cast , reinterpret_cast为什么使用C风格的强制转换可以把想要的任何东西转换成合乎心意的类型.那为什么还需要一个新的C++类型的强制转换呢?新类型的强制转换可以提供更好的控制强制转换过程,允许控制各种不同种类的强制转换.C++中风格是static_cast<type>(content).C++风格的强制转换其他的好处是,它们能更清晰的表明它们要干

  • vue-router中的hash和history两种模式的区别

    众所周知,vue-router有两种模式,hash模式和history模式,这里来谈谈两者的区别. hash模式 hash模式背后的原理是onhashchange事件,可以在window对象上监听这个事件: window.onhashchange = function(event){ console.log(event.oldURL, event.newURL); let hash = location.hash.slice(1); document.body.style.color = has

随机推荐