Skip to main content

Association classification

Basic Concepts

Association objects can be classified from two perspectives, with two types of associations from each perspective, totaling 4 types

  • Classification by association object shape

    • Short Association

      Only modifies the association relationship between the current object and other objects, without further saving the associated objects.

      The recursive saving behavior is terminated and will not continue deeper.

    • Long Association

      Not only modifies the association relationship between the current object and other objects but also further saves the associated objects.

      The recursive saving behavior will not be terminated and will continue deeper.

  • Classification by save order

    • Forward Association

      Associated objects are saved earlier than the current object, which is actually based on foreign keys (whether true or false).

      For example: Book.store in this tutorial.

    • Backward Association

      Associated objects are saved later than the current object, including two situations

      • The inverse association of forward associations.

        For example: BookStore.books in this tutorial

      • Associations based on intermediate tables.

        For example: Book.authors and Author.books in this tutorial

1. Classification by Association Object Shape

1.1. Short Association

A short association means only modifying the association itself between the current object and other objects, with no interest in modifying the associated objects.

Typically, UI design will use radio buttons (association reference) or checkboxes (association collection).

Book Form

 

Where:

  • Radio buttons correspond to many-to-one association Book.store

  • Checkboxes correspond to many-to-many association Book.authors

note

In real projects, when there is a lot of data to choose from, it might not be suitable to design as a dropdown UI. In this case, an object selector with filtering conditions and pagination capabilities can be used instead of a dropdown, which is a common workaround.

Since users only want to modify the association between the current object and other objects, without further modifying the associated objects, the UI cannot have multiple levels of association nesting. This is why it's called a short association.

When passing any shaped data structure as a parameter to the save command, there are two methods to specify short associations:

  • Use Id-Only objects as associated objects

  • Use Key-Only objects as associated objects with special configuration enabled

1.1.1. Using Id-Only Objects as Associated Objects

Make associated objects have only the id property

Book book = Immutables.createBook(draft -> {
draft.setName("SQL in Action");
draft.setEdition(1);
draft.setPrice(new BigDecimal("39.9"));

// Associated object has only id property
draft.setStoreId(2L);

draft.addIntoAuthors(author -> {
// Associated object has only id property
author.setId(4L);
});
draft.addIntoAuthors(author -> {
// Associated object has only id property
author.setId(5L);
});
});
sqlClient.save(book);
note

The hardcoding of the data structure being saved here is only for demonstration. In actual projects, the data structure being saved is submitted by the frontend interface.

Of course, if the user has defined the authorIds property according to the Mapping/Advanced Mapping/View Properties/IdView article, the above code can be simplified, for example:

Book book = ImmutableObjects.createBook(draft -> {
draft.setAuthorIds(Arrays.asList(4L, 5L));
});

However, this is not mandatory. To make the example more universal, we don't assume that users have defined IdView properties for entity types. All subsequent documentation will follow this approach without further reminders.

The generated SQL statements would be:

  1. Save the aggregate root.

    merge into BOOK(
    NAME, EDITION, PRICE, STORE_ID
    ) key(
    NAME, EDITION
    ) values(
    ? /* SQL in Action */,
    ? /* 1 */,
    ? /* 39.9 */,
    ? /* 2 */
    )

    Since Book.store is a many-to-one relationship directly based on foreign key (STORE_ID), the association between the current object and BookStore(2) object will be automatically created by the execution of this SQL.

  2. If there are Author objects associated with the current object (newly inserted data is Book(100)) other than just Author(4) and Author(5), break the associations with those other objects.

    delete from BOOK_AUTHOR_MAPPING
    where
    BOOK_ID = ? /* 100 */
    and
    not (
    AUTHOR_ID = any(? /* [4, 5] */)
    )
    info

    This step is called Dissociation Operation, which will be introduced in subsequent documentation. Readers can ignore this for now.

  3. Establish associations between the object (newly inserted data is Book(100)) and the two objects Author(4) and Author(5)

    merge into BOOK_AUTHOR_MAPPING tb_1_ 
    using(values(?, ?)) tb_2_(
    BOOK_ID, AUTHOR_ID
    )
    on
    tb_1_.BOOK_ID = tb_2_.BOOK_ID
    and
    tb_1_.AUTHOR_ID = tb_2_.AUTHOR_ID
    when not matched
    then insert(BOOK_ID, AUTHOR_ID)
    values(tb_2_.BOOK_ID, tb_2_.AUTHOR_ID)
    /* batch-0: [100, 4] */
    /* batch-1: [100, 5] */
info

Through this example, it's not hard to see that short associations only create or destroy the association relationships between the current object and other objects, without further saving the associated objects.

Short associations always assume that the referenced objects exist. If the referenced objects (in this example, BookStore(2), Author(4), and Author(5)) don't exist, it will cause an exception!

1.1.2. Using Key-Only Objects as Associated Objects with Special Configuration Enabled

In the code below, assume:

  • The key property for the BookStore type is name

  • The key properties for the Author type are firstName and lastName

    In actual business scenarios, this uniqueness constraint might not be reasonable, but let's assume this for the sake of simplifying the example.

Book book = Immutables.createBook(draft -> {
draft.setName("SQL in Action");
draft.setEdition(1);
draft.setPrice(new BigDecimal("39.9"));
draft.applyStore(store -> {
// Associated object has only key property, i.e., `BookStore.name`
store.setName("MANNING");
});
draft.addIntoAuthors(author -> {
// Associated object has only key properties, i.e., `Author.firstName` and `Author.lastName`
author.setFirstName("Boris").setLastName("Cherny");
});
draft.addIntoAuthors(author -> {
// Associated object has only key properties, i.e., `Author.firstName` and `Author.lastName`
author.setFirstName("Samer").setLastName("Buna");
});
});
sqlClient
.saveCommand(book)
.setKeyOnlyAsReference(BookProps.STORE)
.setKeyOnlyAsReference(BookProps.AUTHORS)
.execute();
info

By default, key-only associated objects are treated as long associations

However, developers can treat key-only associated objects as short associations by calling the setKeyOnlyAsReference method.

  • Here, setKeyOnlyAsReference is called twice to explicitly configure the associations Book.store and Book.authors.

    In fact, you can also call setKeyOnlyAsReferenceAll once to blindly configure all associations.

  • Compared to Kotlin, the Java API is slightly less convenient for advanced configuration of save commands.

    First call the saveCommand method to create a save command without executing it immediately, then call the execute method to actually execute it after completing the advanced configuration.

1.2. Long Association

A long association means not only modifying the association itself between the current object and other objects but also further modifying the associated objects.

Typically, orders and order items are the best examples of such scenarios. The UI design will use embedded tables, for example:

CommodityQuantityUnit priceItem priceDelete
14.6929.38
3030
Total price:59.379999999999995
 

Since users not only want to modify the association between the current object and other objects but also want to further modify the associated objects, and the associated objects can contain deeper associations, theoretically, the UI can have multiple levels of association nesting. This is why it's called a long association.

note

Although designers will intentionally avoid nesting deeper embedded tables within embedded tables to maintain UI simplicity, there are still scenarios in real projects where maintaining multi-level nested associations in the UI is necessary, such as:

  • The form itself is a tree structure, which is saved as a whole after editing.

  • Visual UI design, because UI components themselves are tree structures, after users perform a series of visual drag-and-drop designs, the UI component tree is saved as a whole.

Jimmer can directly save long association data structures of any shape. If we call a long association data structure with unknown depth a complex form, the save command is designed for complex forms.

Here's an example:

Order order = Immutables.createOrder(draft -> {
draft.setCustomerId(1L);
draft.setProvince("Prenzlauer Berg");
draft.setCity("Berlin");
draft.setAddress("Brandenburgische Straße 9, Prenzlauer Berg, Berlin, Germany");
draft.addIntoItems(item -> {
item.setProductId(8L);
// Property neither id nor key
item.setQuantity(2);
});
draft.addIntoItems(item -> {
item.setProductId(9L);
// Property neither id nor key
item.setQuantity(1);
});
});
sqlClient.save(order);

In this example, we can see many short associations, such as Order.customer, OrderItem.product, but these are not the focus here.

Here, we should focus on the association Order.items, which is clearly a long association.

This operation generates two SQL statements:

  1. Insert the root object Order

    insert into order_(
    PROVINCE, CITY, ADDRESS, CUSTOMER_ID
    ) values(
    ? /* Prenzlauer Berg */,
    ? /* Berlin */,
    ? /* Brandenburgische Strae 9, Prenzlauer Berg, Berlin, Germany */,
    ? /* 1 */
    )
  2. Insert all child objects OrderItem

    insert into ORDER_ITEM(
    ORDER_ID,
    PRODUCT_ID,
    QUANTITY
    ) values(?, ?, ?)
    /* batch-0: [100, 8, 2] */
    /* batch-1: [100, 9, 1] */
info

As we can see, long associations not only modify the association relationship between the current object and other objects but also cause the associated objects to be saved.

If the associated objects also have long associations, they will be saved recursively until there are no more association properties or until a short association is encountered.

2. Classification by Save Order

2.1. Forward Association

Forward association is based on foreign keys (whether true or false). Its working mode is to save the associated objects first, then save the root object.

Book book = Immutables.createBook(draft -> {
draft.setName("SQL in Action");
draft.setEdition(1);
draft.setPrice(new BigDecimal("49.9"));
draft.applyStore(store -> {
store.setName("TURING");
store.setWebsite("https://www.turing.com");
});
});
sqlClient.save(book);

Taking H2 as an example, it generates two SQL statements:

  1. First save the associated object BookStore

    merge into BOOK_STORE(
    NAME, WEBSITE
    ) key(
    NAME
    ) values(
    ? /* TURING */,
    ? /* https://www.turing.com */
    )
  2. Then save the current object Book

    merge into BOOK(
    NAME, EDITION, PRICE, STORE_ID
    ) key(
    NAME, EDITION
    ) values(
    ? /* SQL in Action */,
    ? /* 1 */,
    ? /* 49.9 */,
    ? /* 100 */
    )
caution

When discussing forward associations in work communication, it's recommended to use expressions like "current object/associated object" rather than "parent object/child object".

This is because, for forward associations, the parent-child relationship at the ORM level is completely opposite to that at the database modeling level, which can easily cause confusion and misunderstanding.

2.2. Backward Association

Other associations, such as:

  • Inverse associations of forward associations (like BookStore.books in this tutorial)
  • Associations based on intermediate tables (like Book.authors and Author.books in this tutorial)

can all be categorized as backward associations, which are more common scenarios.

The working mode of backward associations is easier to understand: save the current object first, then save the associated objects.

BookStore store = Immutables.createBookStore(draft -> {
draft.setName("TURING");
draft.setWebsite("https://www.turing.com");
draft.addIntoBooks(book -> {
book.setName("SQL in Action");
book.setEdition(1);
book.setPrice(new BigDecimal("49.9"));
});
draft.addIntoBooks(book -> {
book.setName("RUST programming");
book.setEdition(2);
book.setPrice(new BigDecimal("39.9"));
});
});
sqlClient
.saveCommand(store)
// Please ignore this configuration for now
.setTargetTransferModeAll(TargetTransferMode.ALLOWED)
.execute();

Taking H2 as an example, it generates three SQL statements:

  1. First save the current object BookStore

    merge into BOOK_STORE(
    NAME, WEBSITE
    ) key(
    NAME
    ) values(
    ? /* TURING */,
    ? /* https://www.turing.com */
    )
  2. Then save the associated objects Book

    merge into BOOK(
    NAME, EDITION, PRICE, STORE_ID
    ) key(
    NAME, EDITION
    ) values(?, ?, ?, ?)
    /* batch-0: [SQL in Action, 1, 49.9, 100] */
    /* batch-1: [RUST programming, 2, 39.9, 100] */
  3. The third SQL statement is not relevant to the topic being discussed here, so it's omitted

info

Backward associations have richer functionality than forward associations. This tutorial will focus on discussing backward associations.