mybatis 忽略字段_MyBatis手把手跟我做系列(二) --- 基本CRUD操作

本文介绍了如何解决MyBatis中数据库字段与Java Bean属性不匹配的问题,包括在全局配置中启用驼峰规则匹配、修改mapper SQL语句以及添加关联映射。此外,还详细讲解了加入log4j以增强日志跟踪,并展示了其他CRUD操作的代码示例,如查询、新增、删除和更新。
摘要由CSDN通过智能技术生成

17864961e056c17046c30aab2c694728.png

在上一章中,已经实现了将数据库表中的内容读取出来,但是结果却是不对的

3219e7703cfbc29d466bd109ed1841d8.png


仔细观察上面的截图,发现数据库t_user表中的字段和User类中的属性完全一样的,数据是赋值进去了的,但是user_tel,registration_time这两个字段在User表中是以驼峰形式命名的属性,找到问题所在,其实数据库字段名和类名中的属性不匹配,引起的问题,这种问题当然很常见,所以Mybatis中有很多种办法来解决这种字段和属性不匹配的问题

一.数据库字段和Java Bean类属性不匹配问题的设置

这个问题我们常见的有三种办法都可以解决

1.在MyBatis全局配置中进行设置,打开驼峰规则匹配

这个也就是在mybatis-configuration.xml文件中,加入下面的配置

<!--开启自动驼峰命名规则(camel case)映射,
    即从经典数据库列名 A_COLUMN 到经典 Java 属性名 aColumn 的类似映射。--><settings>
        <setting name="mapUnderscoreToCamelCase" value="true"/></settings>

注意这段代码的位置,MyBatis的全局配置,对于配置标签的位置是有限制的,不能随便改变设置标签的位置.
现在我们将<settings></settings>标签直接放在<configuration>标签下方

c2c046700fd5ac756daa1c278dc01c23.png

只需要这么设置之后,我们这里的代码就可以正确获取数据库中的值了

d9d43455f21de125b29ec29c02b37d5e.png

<settings></settings>标签中,还有很多我们日常常用的设置

f9229e976454d697d158d58adb1fbecd.png

3d8719b4f1f153b9507899e0d7b1649c.png

afd62d54bc3cf5c8c9046450ccb2dd44.png

083c76e426fc05721c049133c1f48050.png

8ddc1282a3f98e554f538005a97c6541.png

5205edcdad16b2bb19848287d8f0d4be.png

03aa99e129d0a592f6cbd1df7f60a437.png

上面是<settings></settings>中全部的相关配置,不用特意去记住他们,有需要的时候再来查找就行了,其实常用了,也就那几个

2.修改mapper中SQL语句

上面<settings></settings>驼峰规则转换的配置,其实已经能够完美解决我们现在出现的问题,但是如果出现了数据库中字段是username,而java类中的属性是name,那么刚刚配置的转换规则肯定就不匹配了.这种情况也是很常见的,要解决最简单的办法就是直接给数据库字段取一个别名,和java实体类中属性值一致就行了

ccd923474768bf54bbd0504a1555d42b.png

当然,这种只是临时的处理,一般情况下,如果出现字段的不匹配,最常用的是下面的办法

3.添加关联映射

在Mapper文件中添加关联映射,这种方式是处理字段不匹配是最常用的手段

e92176298269312b5a2bbc886c328486.png

1d4fee0ef8985504bf7c7986d74c1937.png

上面的两张截图显示了关联映射应该注意的事情,下面是完整的代码

<?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.yingside.mapper.UserMapper">
    <resultMap id="userMap" type="com.yingside.bean.User">
        <id column="id" property="id" />
        <result column="user_tel" property="userTel"/>
        <result column="username" property="username"/>
        <result column="password" property="password"/>
        <result column="registration_time" property="registrationTime"/>
    </resultMap>

    <select id="getUser" parameterType="int"
            resultMap="userMap">
        select * from t_user where id=#{id}    </select></mapper>

所以,我们在工作中,常常都会使用1,3种办法来处理这种不匹配的情况,一般情况下,驼峰规则映射一下就行了,如果出现特殊情况,这种情况在关联查询的时候还是挺常见的,我们就使用resultMap标签进行一下字段映射

二.加入log4j

如果觉得代码提示过少,我们也可以加入日志,跟踪我们的代码执行过程,使用方式也很简单
1.导入log4j的jar包,这对于我们Maven工程来说,只需要修改POM.xml,加入log4j的包依赖

<!--引入log4j--><dependency>
    <groupId>log4j</groupId>
    <artifactId>log4j</artifactId>
    <version>1.2.17</version></dependency>

2.在Maven工程的resources目录下加入log4j.properties配置文件,在文件中加入log4j相关配置内容

log4j.rootLogger=DEBUG,Console
log4j.logger.com.yingside.mapper=TRACE
log4j.appender.Console=org.apache.log4j.ConsoleAppender
log4j.appender.Console.layout=org.apache.log4j.PatternLayout
log4j.appender.Console.layout.ConversionPattern=%d [%t] %-5p [%c] - %m%n
log4j.logger.org.apache=INFO

就现在这样做的话,我们现在简单的测试工程,已经可以看到log4j的日志效果了

c0f3103f068e8b041c547ecdfdcd4775.png

不过要注意的是,MyBatis内置的日志工厂提供日志功能,具体的日志实现有以下几种工具:

  • SLF4J
  • Apache Commons Logging
  • Log4j 2
  • Log4j
  • JDK logging

具体选择哪个日志实现工具由MyBatis的内置日志工厂确定。它会使用最先找到的(按上文列举的顺序查找)。 如果一个都未找到,日志功能就会被禁用。

我们现在配置了log4j之后,MyBatis顺利找到了log4j,但是不少应用服务器的classpath中已经包含Commons Logging,如我们熟悉的Tomcat, 所以MyBatis会把它作为具体的日志实现。记住这点非常重要。这将意味着,在Tomcat的环境中提供了Commons Logging的私有实现,你的Log4J配置将被忽略。所以,在以后的使用中,我们最好在MyBatis的全局配置的中,再加上log4j日志文件的配置

<configuration>  
  <settings>  
    ...  
    <setting name="logImpl" value="LOG4J"/>  
    ...  
  </settings>  </configuration>

最终目录结构效果:

afbfb8f089c290972fe657dae8fb7e2c.png

三.加入其它CRUD代码

1.查询 <select>

之前只有一个根据id查询对象的方法,现在在mapper文件中加入常见的增删改查的方法,首先加入一个查询所有的方法,查询,在UserMapper.xml配置中,使用的都是<select></select>标签

<!-- 查询 t_user 表中所有用户数据 --><select id="getAll" resultType="com.yingside.bean.User">
    select * from t_user</select>

如果我们这么写了之后,你会发现打印的效果其实和之前一样,userTel和registrationTime得到的结果是null,其实原因是一样的,这是最好还是应该使用resultMap

f0ea8ea34085ca91134736f630c9d516.png

还是在测试文件中测试一下效果,在Test中添加testGetAllUsers()方法

@org.junit.Testpublic void testGetAllUsers(){
    String stmt = "com.yingside.mapper.UserMapper.getAll";
    List<User> users = sqlSession.selectList(stmt);
    log.info(users);
    sqlSession.close();
}

85cb6ebad3c4606f5c4401426e4e8a59.png

2.新增 <insert>

先在UserMapper.xml中加入对应的SQL语句标签,新增语句使用<insert></insert>标签

注意新增语句肯定是要带参数的,而且参数一般都很多,所以直接以面向对象的方式插入一个User对象,然后在sql语句中调用的都是对象的属性值,在insert标签中,使用parameterType属性表示要新增对象的类型

UserMapper.xml:

<!-- 向 t_user 表插入一条数据 --><insert id="insertUser" parameterType="com.yingside.bean.User">
    insert into t_user(id,user_tel,username,password,registration_time)
        value(null,#{userTel},#{username},#{password},#{registrationTime})</insert>

Test.java:

@org.junit.Testpublic void testInsertUser(){
    String stmt = "com.yingside.mapper.UserMapper.insertUser";
    User user = new User();
    user.setUsername("韦小宝");
    user.setPassword("aaaaaa");
    user.setUserTel("13880000008");
    user.setRegistrationTime("2018-12-12");    int n = sqlSession.insert(stmt,user);
    log.info(n);
    sqlSession.close();
}

e89ad5c9836fd93e1173f542231c4f4c.png

上图表明了整个查询执行的过程和结果

2.1. 新增数据返回主键id值

不过这里的结果是一般数据库DML语句表示几行受影响的结果,一般我们的新增操作要求都比较特殊,需要知道最新插入的数据主键id是多少,如果这样的话,可以在<insert>标签中加入下面的代码

<!-- 向 t_user 表插入一条数据 --><insert id="insertUser" parameterType="com.yingside.bean.User">
    <!--将插入的数据主键返回到 user 对象中
       keyProperty:将查询到的主键设置到parameterType 指定到对象的那个属性
       select LAST_INSERT_ID():查询上一次执行insert 操作返回的主键id值,只适用于自增主键
       resultType:指定 select LAST_INSERT_ID() 的结果类型
       order:AFTER,相对于 select LAST_INSERT_ID()操作的顺序
    -->
    <selectKey keyProperty="id" resultType="int" order="AFTER">
        select LAST_INSERT_ID()    </selectKey>
    insert into t_user(id,user_tel,username,password,registration_time)
        value(null,#{userTel},#{username},#{password},#{registrationTime})</insert>

当然上面的代码完全可以简写成下面这种方式

<insert id="insertUser" parameterType="user" useGeneratedKeys="true" keyProperty="id"></insert>

重新测试一下看看执行效果

ebd8431dd3b2eea709eeb039a24fd436.png

mysql数据库是支持主键自增id的,上面这种办法,也只是主键自增id的写法,如果是非自增主键机制,比如使用UUID,就需要换一种写法

<insert id="insertUser" parameterType="com.yingside.bean.User">
    <!-- 将插入的数据主键返回到 user 对象中
         首先通过 select UUID()得到主键值,然后设置到 user 对象的id中,再进行 insert 操作
         keyProperty: 将查询到的主键设置到parameterType 指定到对象的那个属性
         select UUID():得到主键的id值,注意这里是字符串
         resultType: 指定 select UUID() 的结果类型
         order: BEFORE,相对于 select UUID()操作的顺序
         注意先后顺序和自增主键机制是不一样的
     -->
    <selectKey keyProperty="id" resultType="String" order="BEFORE">
        select UUID()    </selectKey>
    insert into t_user(id,user_tel,username,password,registration_time)
        value(#{id},#{userTel},#{username},#{password},#{registrationTime})</insert>

3.删除 <delete>

删除使用<delete>标签,一般情况下,我们删除都根据主键id进行删除

UserMapper.xml:

<!-- 根据 id 删除 user 表的数据 --><delete id="deleteUserById" parameterType="int">
    delete from t_user where id=#{id}</delete>

Test.java:

@org.junit.Testpublic void testDeleteUser(){
    String stmt = "com.yingside.mapper.UserMapper.deleteUserById";    int id = 13;    int n = sqlSession.delete(stmt,id);    //提交新增的数据
    sqlSession.commit();
    log.info(n);
    sqlSession.close();
}

e24b38339c3e341522803c102252179f.png

4.更新 <update>

更新使用<update>标签,一般情况下,是根据主键id,更新数据

UserMapper.xml:

<!-- 根据 id 更新 user 表的数据 --><update id="updateUserById" parameterType="com.yingside.bean.User">
  update t_user set user_tel=#{userTel},username=#{username},
  password=md5(#{password}),registration_time=#{registrationTime}
  where id=#{id}</update>

Test.java:

@org.junit.Testpublic void testUpdateUser(){
    String stmt = "com.yingside.mapper.UserMapper.updateUserById";
    User user = new User();
    user.setId(8);
    user.setUsername("令狐冲");
    user.setPassword("abcdefg");
    user.setUserTel("13880000009");
    user.setRegistrationTime("2018-12-20");    int n = sqlSession.insert(stmt,user);    //提交修改的数据
    sqlSession.commit();
    sqlSession.close();
}

47f3c9f1783c017896a540e5c308208d.png

四.配置别名<typeAliases>

这就是最基本的数据库CRUD操作,不过上面的xml配置中,有一个的重复很烦躁,那就是 parameterType="com.yingside.bean.User"如果是自定义的类型,就必须跟上类的包名全路径,那这个其实只需要在全局环境中配置一下别名就可以了

<configuration>
   ...    <typeAliases>
        <!--给com.yingside.bean包下的所有类起一个别名,默认就是类名 com.yingside.bean.User === User
        (注意低版本的MyBatis 别名默认是类名首字母小写,相当于 com.yingside.bean.User === user)
        这样在每个mapper文件中如果出现自定义的类,就不必再写成 包路径 + 类名 的形式了,直接写这里定义的别名就ok
        -->
        <package name="com.yingside.bean"/>
        <!--单独的给每个类别名-->
        <!--<typeAlias type="com.yingside.bean.User" alias="user" />-->
    </typeAliases>
    ... </configuration>

31f1e6d49a3402820f13ac9b7415275e.png

原文转自朗沃易课堂,更多详情精彩可以搜索“朗沃”公众号关注更多!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值