2.2 数据库映射

在 2.1 中看到的是最简单的情况,实际使用过程中也不会更复杂,下面是更详细的映射配置。

通用 Mapper 中,默认情况下是将实体类字段按照驼峰转下划线形式的表名列名进行转换。

例如

实体类的 userName 可以映射到表的 user_name 上。

如果想要修改默认的转换方式,可以在后续的配置中,修改 style 全局配置。

数据库映射主要涉及到一些注解和全局配置,这一节中会介绍所有注解,后面章节会有配置的介绍。

通用 Mapper 默认使用了几个简单的注解,其他 JPA 的注解默认并不支持,但是如果你开发自己的通用方法,你可以使用 JPA 注解或者引入自己的注解。

2.2.1@NameStyle 注解(Mapper)

这个注解可以在类上进行配置,优先级高于对应的 style 全局配置。

注解支持以下几个选项:

  1. normal, //原值
  2. camelhump, //驼峰转下划线
  3. uppercase, //转换为大写
  4. lowercase, //转换为小写
  5. camelhumpAndUppercase, //驼峰转下划线大写形式
  6. camelhumpAndLowercase, //驼峰转下划线小写形式

使用时,直接在类上配置即可,例如:

  1. @NameStyle(Style.camelhumpAndUppercase)
  2. public class Country

配置该注解后,对该类和其中的字段进行转换时,会将形如 userName 的字段转换为表中的 USER_NAME 字段。

2.2.2 @Table 注解(JPA)

@Table 注解可以配置 name,catalogschema 三个属性,配置 name 属性后,直接使用提供的表名,不再根据实体类名进行转换。其他两个属性中,同时配置时,catalog 优先级高于 schema,也就是只有 catalog 会生效。

配置示例如下:

  1. @Table(name = "sys_user")
  2. public class User

User 实体映射到 sys_user 表。

2.2.3 @Column 注解(JPA)

@Column 注解支持 name, insertableupdateable 三个属性。

name 配置映射的列名。

insertable 对提供的 insert 方法有效,如果设置 false 就不会出现在 SQL 中。

updateable 对提供的 update 方法有效,设置为 false 后不会出现在 SQL 中。

配置示例如:

  1. @Column(name = "user_name")
  2. private String name;

除了直接映射 nameuser_name 这种用法外,在使用关键字的情况,还会有下面的用法:

  1. @Column(name = "`order`")
  2. private String order;

对于关键字这种情况,通用 Mapper 支持自动转换,可以查看后续配置文档中的 wrapKeyword 配置。

2.2.4 @ColumnType 注解(Mapper)

这个注解提供的 column属性和 @Column 中的 name 作用相同。但是 @Column 的优先级更高。

除了 name 属性外,这个注解主要提供了 jdbcType 属性和 typeHandler 属性。

jdbcType 用于设置特殊数据库类型时指定数据库中的 jdbcType

typeHandler 用于设置特殊类型处理器,常见的是枚举。

用法示例如下:

  1. @ColumnType(
  2. column = "countryname",
  3. jdbcType = JdbcType.VARCHAR,
  4. typeHandler = StringTypeHandler.class)
  5. private String countryname;

2.2.5 @Transient 注解(JPA)

一般情况下,实体中的字段和数据库表中的字段是一一对应的,但是也有很多情况我们会在实体中增加一些额外的属性,这种情况下,就需要使用 @Transient 注解来告诉通用 Mapper 这不是表中的字段。

默认情况下,只有简单类型会被自动认为是表中的字段(可以通过配置中的 useSimpleType 控制)。

这里的简单类型不包含 Java 中的8种基本类型:

byte,short,int,long,float,double,char,boolean

这是因为在类中,基本类型会有默认值,而 MyBatis 中经常会需要判断属性值是否为空,所以不要在类中使用基本类型,否则会遇到莫名其妙的错误。

对于类中的复杂对象,以及 Map,List 等属性不需要配置这个注解。

对于枚举类型作为数据库字段的情况,需要看配置中的 enumAsSimpleType 参数。

配置示例:

  1. @Transient
  2. private String otherThings; //非数据库表中字段

2.2.6 @Id 注解(JPA)

上面几个注解都涉及到映射。 @Id 注解和映射无关,它是一个特殊的标记,用于标识数据库中的主键字段。

正常情况下,一个实体类中至少需要一个标记 @Id 注解的字段,存在联合主键时可以标记多个。

如果表中没有主键,类中就可以不标记。

当类中没有存在标记 @Id 注解的字段时,你可以理解为类中的所有字段是联合主键。使用所有的 ByPrimaryKey 相关的方法时,有 where 条件的地方,会将所有列作为条件。

配置示例:

  1. @Id
  2. private Integer id;

或者联合主键:

  1. @Id
  2. private Integer userId;
  3. @Id
  4. private Integer roleId;

2.2.7 @KeySql 注解

主键策略注解,用于配置如何生成主键。

这是通用 Mapper 的自定义注解,改注解的目的就是替换 @GeneratedValue 注解。

关于该注解的用法可以查看 2.3 主键策略

2.2.8 @GeneratedValue 注解(JPA)

主键策略注解,用于配置如何生成主键。

由于不同类型数据库的配置不同,所以后面有一节专门介绍该注解的文档。

关于该注解的用法可以查看 2.3 主键策略

推荐使用上面的 @KeySql 注解。

2.2.9 @Version 注解(Mapper)

@Version 是实现乐观锁的一个注解,大多数人都不需要。

需要使用该注解的请看 2.4 乐观锁

2.2.10 @RegisterMapper 注解

为了解决通用 Mapper 中最常见的一个错误而增加的标记注解,该注解仅用于开发的通用接口,不是实体类上使用的,这里和其他注解一起介绍了。

4.0 版本提供的所有通用接口上都标记了该注解,因此自带的通用接口时,不需要配置 mappers 参数,该注解的具体用法会在 第五章 扩展通用接口 中介绍。

2.2.11 自定义注解

这部分提供给想要开发通用方法的朋友,如果只是使用,不需要阅读这部分内容。

在通用 Mapper 中,可以通过 EntityHelper.getColumns(entityClass) 方法来获取实体类的全部信息。

EntityColumn 中,通过下面的代码可以获取字段上的任意注解。

  1. //判断是否有某个注解
  2. boolean hasVersion = column.getEntityField().isAnnotationPresent(Version.class)
  3. //通过下面的代码可以获取注解信息
  4. Version version = column.getEntityField().getAnnotation(Version.class);

通过这种方式,在实现自己的通用方式时,可以根据需要来增加额外的注解来实现一些其他的用途。