跳到主要内容

定义实体

和以JPA/Hibernate为代表的ORM不同,Jimmer中的实体定义采用interface,而非class。原因在下一节讨论。

在定义实体之前,需要声明两个概念:

  • Jimmer中实体对象并非简单的Java Bean,而是动态对象。

    即,某个属性未被设置和某个属性被设置为null,是完全不同的。

  • Jimmer实体是不可变对象,因此,接口中指存在getter,不存在setter。

定义非关联字段

假设实体所在的包为"com.example.model",先忽略关联属性,各实体定义如下。

  • BookStore

    BookStore.java
    package com.example.model;

    import org.babyfish.jimmer.sql.*;
    import org.jetbrains.annotations.Nullable;

    @Entity
    public interface BookStore {

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    long id();

    @Key
    String name();

    @Nullable
    String website();
    }
  • Book

    Book.java
    package com.example.model;

    import org.babyfish.jimmer.sql.*;
    import java.math.BigDecimal;

    @Entity
    public interface Book {

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    long id();

    @Key
    String name();

    @Key
    int edition();

    BigDecimal price();
    }
  • Author

    Author.java
    package com.example.model;

    import org.babyfish.jimmer.sql.*;

    @Entity
    public interface Author {

    @Id
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    long id();

    @Key
    String firstName();

    @Key
    String lastName();

    /*
    * 这里,Gender是一个枚举,代码稍后给出
    */
    Gender gender();
    }

    其中Gender是一个枚举,只有两个选项:MALEFEMALE

    ORM处理枚举有两种方式:

    • 映射为字符串:可观察性优先的选择,也是默认的选择。
    • 映射为数字:性能优先的选择。

    虽然本教程示例按照默认方式把枚举映射成字符串,但数据库中check约束限定的取值是MF,并非默认的MALEFEMALE,所以,这个枚举需要配置如下。

    Gender.java
    package com.example.model;

    import org.babyfish.jimmer.sql.EnumItem;

    public enum Gender {

    @EnumItem(name = "M")
    MALE,

    @EnumItem(name = "F")
    FEMALE
    }
    信息

    完整的自定义标量类型的映射,请参见TODO

  • TreeNode

    TreeNode.java
    package com.example.model;

    import org.babyfish.jimmer.sql.*;

    @Entity
    public interface TreeNode {

    @Id
    @Column(name = "NODE_ID")
    @GeneratedValue(strategy = GenerationType.IDENTITY)
    long id();

    String name();
    }
  • 对于Java而言,每一个getter,既可以采用传统Java Bean那种get/is开头的方方式,比如getName(),也可以采用当前例子中不用get/is开头的方式,比如name()

    备注

    打破传统Java Bean中getter方法必须以get/is开头的规则的并非Jimmer,而是Java14引入的record类型。这种新的风格可以让不可变对象的书写变得更简单。

  • Jimmer实体对每个属性是否为null非常敏感。

    • 对于kotlin而言,采用语言本身的nullity描述。
    • 对于Java而言:
      • boolean, char, byte, short, int, long, float, double表示不能为null
      • Boolean, Character, Byte, Short, Integer, Long, Float, Double表示允许为null
      • 其他类型默认不能为null,如果期望允许为null,请加注解@org.jetbrains.annotations.Nullable@javax.validation.constraints.Null@org.springframework.lang.Nullable
  • 例子使用的注解:

    • @org.babyfish.jimmer.sql.Entity: 指定实体类型。
    • @org.babyfish.jimmer.sql.Id: 指定id属性。
    • @org.babyfish.jimmer.sql.GeneratedValue: 指定id自动分配策略,这里使用数据库的自动编号。
  • 代码中Java/Kotlin接口的短名,按照word1Word2...WordN -> WORD1_WORD2_..._WORDN的规则转换后,和数据库中对应的表名完全匹配。

    如果Java/Kotlin接口的短名和数据库表名无法自动匹配,请用@org.babyfish.jimmer.sql.Table修饰接口。

  • 同理,代码中Java/Kotlin属性,按照上述规则转换后,和数据库中对应的列名完全匹配。

    如果Java/Kotlin属性名和数据库列名无法自动匹配,请用@org.babyfish.jimmer.sql.Column修饰属性。

    备注

    注意:该注解仅可用于修饰非关联字段,而非下文要讨论的外键字段

  • Jimmer实体是动态对象,没有id的对象,就是id未被设置的对象,而非id被设置为null的对象。Jimmer不需要让id可以为null。

    所以,代码中id属性的类型为不能为null的long,而非可以为null的Long,否则框架会报错提示。

    备注

    有JPA/Hibernate背景知识的读者需注意这个细节。

多对一关联Book.store

修改Book.java/Book.kt,添加一个关联属性store

Book.java
package com.example.model;

...省略导入语句...

@Entity
public interface Book {

...省略其他属性...

@ManyToOne
@Nullable
BookStore store();
}
  • @org.babyfish.jimmer.sql.ManyToOne声明多对一关联属性,将数据库的外键字段映射为关联对象。

  • 代码中Java/Kotlin属性,按照word1Word2...WordN -> WORD1_WORD2_..._WORDN_ID的规则转换,即store -> STORE_ID,结果和数据库中外键的列名完全匹配。

    如果Java/Kotlin属性名和数据库列名无法自动匹配,请用@org.babyfish.jimmer.sql.JoinColumn修饰属性。

一对多关联BookStore.books

修改BookStore.java/BookStore.kt,添加一个关联属性books

BookStore.java
package com.example.model;

...省略导入语句...

@Entity
public interface BookStore {

...省略其他属性...

@OneToMany(mappedBy = "store")
List<Book> books();
}
  • @org.babyfish.jimmer.sql.OneToMany声明对一对多关联。

    一对多关联并不映射数据库中的任何字段,它仅用于对另一个多对一关联做镜像映射,将实体之间单向关联变成双向关联。

    这里,@ManyToOne(mappedBy = "store"),指当前属性BookStore.booksBook.store的镜像。

    @Entity
    public interface BookStore {

    @OneToMany(
    mappedBy = "store"
    )
    List<Book> books();

    ...
    }

    mirror

    @Entity
    public interface Book {

    @ManyToOne
    @Nullable
    BookStore store();

    ...
    }

    关联注解具备mappedBy的属性(这里的BookStore.books),叫做镜像属性。

  • 镜像属性是可选的,因此,双向关联不是必须的。

  • JPA/Hibernate不同,Jimmer中的一对多关联只能作为多对一关联的镜像。即,一对多关联一定意味着双向关联。

多对多关联Book.authors

修改Book.java/Book.kt,添加一个关联属性authors

Book.java
package com.example.model;

...省略导入语句...

@Entity
public interface Book {

...省略其他属性...

@ManyToMany
@JoinTable(
name = "BOOK_AUTHOR_MAPPING",
joinColumnName = "BOOK_ID",
inverseJoinColumnName = "AUTHOR_ID"
)
List<Author> authors();
}
  • @org.babyfish.jimmer.sql.ManyToMany声明对多对多关联。

  • 多对多关联分为两种,主动端和镜像端,这里的@ManyToMany是主动端。镜像端在后文介绍。

  • 对于主动端的多对多关联,可以使用@org.babyfish.jimmer.sql.JoinTable设置中间表

    • name: 中间表的表名

      如果不指定,默认为:{当前实体短名驼峰转下划线}_{目标实体短名驼峰转下划线}_MAPPING

    • joinColumnName: 中间表引用当前实体类型(这里是Book)的外键

      如果不指定,默认为:{当前实体短名驼峰转下划线}_ID

    • inverseJoinColumnName: 中间表引用目标实体类型(这里是Author)的外键

      如果不指定,默认为:{目标实体短名驼峰转下划线}_ID

    备注

    不难看出,这个例子中,@JoinTable的name,joinColumnName和inverseJoinColumnName的值都被设置成了默认值。 这种情况下,其实可以省略这个@JoinTable的。

    但是,为了保持必要的代码可读性,还是写了这个@JoinTable

多对多关联Author.books

修改Author.java/Author.kt,添加一个关联属性books

Author.java
package com.example.model;

...省略导入语句...

@Entity
public interface Author {

...省略其他属性...

@ManyToMany(mappedBy = "authors")
List<Book> books();
}

这里,@org.babyfish.jimmer.sql.ManyToMany注解的mappedBy被设置了,这表示是一个镜像端的多对多映射。

@ManyToMany(mappedBy = "authors"),指当前属性Author.booksBook.authors的镜像。

@Entity
public interface Author {

@ManyToMany(
mappedBy = "authors"
)
List<Book> books();

...
}

mirror

@Entity
public interface Book {

@ManyToMany
@JoinTable(......)
List<Author> authors();

...
}

镜像端的多对多关联是可选的,即,双向多对多关联是可选的。但是,一旦声明双向多对多关联,必须一端为主动端,一端是镜像端。

提示

和JPA/Hibernate不同,在JPA/Hibernate中,主动端和镜像端的抉择非常重要,这会对数据保存行为的编程方式造成影响。

Jimmer没有这个问题,你可以随意决定,没有任何影响。

完善TreeNode的定义

现在,读者应该理解了关联属性。让我们快速完善TreeNode的定义:

TreeNode.java
package com.example.model;

...省略导入语句...

@Entity
public interface TreeNode {

...省略其他属性...

@ManyToOne
@Nullable
TreeNode parent();

@OneToMany(mappedBy = "parent")
List<TreeNode> childNodes();
}

为什么实体是interface

在前面的例子中,我们看到Jimmer中实体类型声明采用interface而非class,这是为什么呢?

  1. 动态性

    Jimmer实体并非简单的Java Bean,而是动态对象。

    • 某个属性未被设置和某个属性被设置为null,是完全不同的。

    • 直接访问某个不存在的属性,异常。

    • 使用Jackson序列化动态对象,不会异常,而是自动忽略未设置属性。

    完整的动态性论述,请参见动态性

    • 动态性的好处

      • 轻松表达任意复杂的数据结构。任何一个实体对象,都有无限的表达力,可能是一个残缺的对象,也可能是一个完整的对象,还可能是一个复杂树的聚合根。

      • 既然可以轻松表达任意复杂的数据结构,那么,ORM就可以以整个数据结构为基本操作单位,而非以单个实体对象为操作单位,对整个数据结构实现一句话查询、一句话保存。

      • 对于查询业务而言,描述任意复杂的数据结构可以直接作为HTTP服务的返回,无需为实现每一种特定形状的返回数据结构不厌其烦地定义DTO类型并逐一映射。

    • Jimmer和动态语言方案的对比

      如上所述,Jimmer利用实体对象的动态性,提供静态语言ORM难以想象的灵活性。

      然而,Jimmer拒绝动态语言高度的不安全性和不可维护性。Jimmer实体仍然是普通的Java/Kotlin对象,保住了静态语言的拼写安全、类型安全、甚至空安全(限kotlin)。

      备注

      Jimmer的动态实体唯一的不安全问题是,直接访问不存在的属性会导致异常:org.babyfish.jimmer.UnloadedException

      这个问题并非Jimmer独有的问题,Jimmer对象的动态性其实是把JPA/Hibernate中的延迟属性推广到了任何属性,这个异常本质上和org.hibernate.LazyInitializationException相似。

      即,这是ORM领域早已普遍接受的必要成本。

    提示

    Jimmer的本质,是在动态语言ORM的灵活性和静态语言的安全性之间,寻找最佳折中和完美平衡点。

  2. 不可变性

    从前文的代码中,我们看到实体接口中只有getter没有setter,这表示实体对象是不可变的。

    可变数据对象纵容了环形引用,这是业务系统中最头疼的问;不可变对象很好地规避了这个问题。

    然而,享受并利用不可变对象的优势,压制其的劣势,并不是一件容易的事情。尤其是对ORM这类对象层次结构很深的场景。

    幸运的是,在JavaScript/TypeScript领域,有一个叫做immer的框架,给出一套完美的深层次不可变对象的处理方法。

    Jimmer为Java/Kotlin移植了immer,让其成为整个ORM的基础编程模型,项目取名"Jimmer"也为致敬immer

    如何创建和“修改”不可变对象的细节,请参考不可变性/方案

综上,由于Jimmer对象本身具备动态性且不可变,所以它并非简单的Java对像,其类型有着复杂的内部实现。

因此,Jimmer选择让开发人员书写interface,并由AnnotationProcessor(java)或KSP(kotlin)在编译时生成它们的实现。