What is “the inverse side of the association” in a bidirectional JPA OneToMany/ManyToOne association? What is “the inverse side of the association” in a bidirectional JPA OneToMany/ManyToOne association? java java

What is “the inverse side of the association” in a bidirectional JPA OneToMany/ManyToOne association?


To understand this, you must take a step back. In OO, the customer owns the orders (orders are a list in the customer object). There can't be an order without a customer. So the customer seems to be the owner of the orders.

But in the SQL world, one item will actually contain a pointer to the other. Since there is 1 customer for N orders, each order contains a foreign key to the customer it belongs to. This is the "connection" and this means the order "owns" (or literally contains) the connection (information). This is exactly the opposite from the OO/model world.

This may help to understand:

public class Customer {     // This field doesn't exist in the database     // It is simulated with a SQL query     // "OO speak": Customer owns the orders     private List<Order> orders;}public class Order {     // This field actually exists in the DB     // In a purely OO model, we could omit it     // "DB speak": Order contains a foreign key to customer     private Customer customer;}

The inverse side is the OO "owner" of the object, in this case the customer. The customer has no columns in the table to store the orders, so you must tell it where in the order table it can save this data (which happens via mappedBy).

Another common example are trees with nodes which can be both parents and children. In this case, the two fields are used in one class:

public class Node {    // Again, this is managed by Hibernate.    // There is no matching column in the database.    @OneToMany(cascade = CascadeType.ALL) // mappedBy is only necessary when there are two fields with the type "Node"    private List<Node> children;    // This field exists in the database.    // For the OO model, it's not really necessary and in fact    // some XML implementations omit it to save memory.    // Of course, that limits your options to navigate the tree.    @ManyToOne    private Node parent;}

This explains for the "foreign key" many-to-one design works. There is a second approach which uses another table to maintain the relations. That means, for our first example, you have three tables: The one with customers, the one with orders and a two-column table with pairs of primary keys (customerPK, orderPK).

This approach is more flexible than the one above (it can easily handle one-to-one, many-to-one, one-to-many and even many-to-many). The price is that

  • it's a bit slower (having to maintain another table and joins uses three tables instead of just two),
  • the join syntax is more complex (which can be tedious if you have to manually write many queries, for example when you try to debug something)
  • it's more error prone because you can suddenly get too many or too few results when something goes wrong in the code which manages the connection table.

That's why I rarely recommend this approach.


Unbelievably, in 3 years nobody has answered your excellent question with examples of both ways to map the relationship.

As mentioned by others, the "owner" side contains the pointer (foreign key) in the database. You can designate either side as the owner, however, if you designate the One side as the owner, the relationship will not be bidirectional (the inverse aka "many" side will have no knowledge of its "owner"). This can be desirable for encapsulation/loose coupling:

// "One" Customer owns the associated orders by storing them in a customer_orders join tablepublic class Customer {    @OneToMany(cascade = CascadeType.ALL)    private List<Order> orders;}// if the Customer owns the orders using the customer_orders table,// Order has no knowledge of its Customerpublic class Order {    // @ManyToOne annotation has no "mappedBy" attribute to link bidirectionally}

The only bidirectional mapping solution is to have the "many" side own its pointer to the "one", and use the @OneToMany "mappedBy" attribute. Without the "mappedBy" attribute Hibernate will expect a double mapping (the database would have both the join column and the join table, which is redundant (usually undesirable)).

// "One" Customer as the inverse side of the relationshippublic class Customer {    @OneToMany(cascade = CascadeType.ALL, mappedBy = "customer")    private List<Order> orders;}// "many" orders each own their pointer to a Customerpublic class Order {    @ManyToOne    private Customer customer;}


The entity which has the table with foreign key in the database is the owning entity and the other table, being pointed at, is the inverse entity.