pages/docs/associations.md
GORM automates the saving of associations and their references when creating or updating records, using an upsert technique that primarily updates foreign key references for existing associations.
When you create a new record, GORM will automatically save its associated data. This includes inserting data into related tables and managing foreign key references.
user := User{
Name: "jinzhu",
BillingAddress: Address{Address1: "Billing Address - Address 1"},
ShippingAddress: Address{Address1: "Shipping Address - Address 1"},
Emails: []Email{
{Email: "[email protected]"},
{Email: "[email protected]"},
},
Languages: []Language{
{Name: "ZH"},
{Name: "EN"},
},
}
// Creating a user along with its associated addresses, emails, and languages
db.Create(&user)
// BEGIN TRANSACTION;
// INSERT INTO "addresses" (address1) VALUES ("Billing Address - Address 1"), ("Shipping Address - Address 1") ON DUPLICATE KEY DO NOTHING;
// INSERT INTO "users" (name,billing_address_id,shipping_address_id) VALUES ("jinzhu", 1, 2);
// INSERT INTO "emails" (user_id,email) VALUES (111, "[email protected]"), (111, "[email protected]") ON DUPLICATE KEY DO NOTHING;
// INSERT INTO "languages" ("name") VALUES ('ZH'), ('EN') ON DUPLICATE KEY DO NOTHING;
// INSERT INTO "user_languages" ("user_id","language_id") VALUES (111, 1), (111, 2) ON DUPLICATE KEY DO NOTHING;
// COMMIT;
db.Save(&user)
FullSaveAssociationsFor scenarios where a full update of the associated data is required (not just the foreign key references), the FullSaveAssociations mode should be used.
// Update a user and fully update all its associations
db.Session(&gorm.Session{FullSaveAssociations: true}).Updates(&user)
// SQL: Fully updates addresses, users, emails tables, including existing associated records
Using FullSaveAssociations ensures that the entire state of the model, including all its associations, is reflected in the database, maintaining data integrity and consistency throughout the application.
GORM provides flexibility to skip automatic saving of associations during create or update operations. This can be achieved using the Select or Omit methods, which allow you to specify exactly which fields or associations should be included or excluded in the operation.
Select to Include Specific FieldsThe Select method lets you specify which fields of the model should be saved. This means that only the selected fields will be included in the SQL operation.
user := User{
// User and associated data
}
// Only include the 'Name' field when creating the user
db.Select("Name").Create(&user)
// SQL: INSERT INTO "users" (name) VALUES ("jinzhu");
Omit to Exclude Fields or AssociationsConversely, Omit allows you to exclude certain fields or associations when saving a model.
// Skip creating the 'BillingAddress' when creating the user
db.Omit("BillingAddress").Create(&user)
// Skip all associations when creating the user
db.Omit(clause.Associations).Create(&user)
{% note warn %}
NOTE:
For many-to-many associations, GORM upserts the associations before creating join table references. To skip this upserting, use Omit with the association name followed by .*:
// Skip upserting 'Languages' associations
db.Omit("Languages.*").Create(&user)
To skip creating both the association and its references:
// Skip creating 'Languages' associations and their references
db.Omit("Languages").Create(&user)
{% endnote %}
Using Select and Omit, you can fine-tune how GORM handles the creation or updating of your models, giving you control over the auto-save behavior of associations.
In GORM, when creating or updating records, you can use the Select and Omit methods to specifically include or exclude certain fields of an associated model.
With Select, you can specify which fields of an associated model should be included when saving the primary model. This is particularly useful for selectively saving parts of an association.
Conversely, Omit lets you exclude certain fields of an associated model from being saved. This can be useful when you want to prevent specific parts of an association from being persisted.
user := User{
Name: "jinzhu",
BillingAddress: Address{Address1: "Billing Address - Address 1", Address2: "addr2"},
ShippingAddress: Address{Address1: "Shipping Address - Address 1", Address2: "addr2"},
}
// Create user and his BillingAddress, ShippingAddress, including only specified fields of BillingAddress
db.Select("BillingAddress.Address1", "BillingAddress.Address2").Create(&user)
// SQL: Creates user and BillingAddress with only 'Address1' and 'Address2' fields
// Create user and his BillingAddress, ShippingAddress, excluding specific fields of BillingAddress
db.Omit("BillingAddress.Address2", "BillingAddress.CreatedAt").Create(&user)
// SQL: Creates user and BillingAddress, omitting 'Address2' and 'CreatedAt' fields
GORM allows for the deletion of specific associated relationships (has one, has many, many2many) using the Select method when deleting a primary record. This feature is particularly useful for maintaining database integrity and ensuring related data is appropriately managed upon deletion.
You can specify which associations should be deleted along with the primary record by using Select.
// Delete a user's account when deleting the user
db.Select("Account").Delete(&user)
// Delete a user's Orders and CreditCards associations when deleting the user
db.Select("Orders", "CreditCards").Delete(&user)
// Delete all of a user's has one, has many, and many2many associations
db.Select(clause.Associations).Delete(&user)
// Delete each user's account when deleting multiple users
db.Select("Account").Delete(&users)
{% note warn %} NOTE: It's important to note that associations will be deleted only if the primary key of the deleting record is not zero. GORM uses these primary keys as conditions to delete the selected associations.
// This will not work as intended
db.Select("Account").Where("name = ?", "jinzhu").Delete(&User{})
// SQL: Deletes all users with name 'jinzhu', but their accounts won't be deleted
// Correct way to delete a user and their account
db.Select("Account").Where("name = ?", "jinzhu").Delete(&User{ID: 1})
// SQL: Deletes the user with name 'jinzhu' and ID '1', and the user's account
// Deleting a user with a specific ID and their account
db.Select("Account").Delete(&User{ID: 1})
// SQL: Deletes the user with ID '1', and the user's account
{% endnote %}
Association Mode in GORM offers various helper methods to handle relationships between models, providing an efficient way to manage associated data.
To start Association Mode, specify the source model and the relationship's field name. The source model must contain a primary key, and the relationship's field name should match an existing association.
var user User
db.Model(&user).Association("Languages")
// Check for errors
error := db.Model(&user).Association("Languages").Error
Retrieve associated records with or without additional conditions.
// Simple find
db.Model(&user).Association("Languages").Find(&languages)
// Find with conditions
codes := []string{"zh-CN", "en-US", "ja-JP"}
db.Model(&user).Where("code IN ?", codes).Association("Languages").Find(&languages)
Add new associations for many to many, has many, or replace the current association for has one, belongs to.
// Append new languages
db.Model(&user).Association("Languages").Append([]Language{languageZH, languageEN})
db.Model(&user).Association("Languages").Append(&Language{Name: "DE"})
db.Model(&user).Association("CreditCard").Append(&CreditCard{Number: "411111111111"})
Replace current associations with new ones.
// Replace existing languages
db.Model(&user).Association("Languages").Replace([]Language{languageZH, languageEN})
db.Model(&user).Association("Languages").Replace(Language{Name: "DE"}, languageEN)
Remove the relationship between the source and arguments, only deleting the reference.
// Delete specific languages
db.Model(&user).Association("Languages").Delete([]Language{languageZH, languageEN})
db.Model(&user).Association("Languages").Delete(languageZH, languageEN)
Remove all references between the source and association.
// Clear all languages
db.Model(&user).Association("Languages").Clear()
Get the count of current associations, with or without conditions.
// Count all languages
db.Model(&user).Association("Languages").Count()
// Count with conditions
codes := []string{"zh-CN", "en-US", "ja-JP"}
db.Model(&user).Where("code IN ?", codes).Association("Languages").Count()
Association Mode allows you to handle relationships for multiple records in a batch. This includes finding, appending, replacing, deleting, and counting operations for associated data.
db.Model(&users).Association("Role").Find(&roles)
db.Model(&users).Association("Team").Delete(&userA)
db.Model(&users).Association("Team").Count()
var users = []User{user1, user2, user3}
// Append different teams to different users in a batch
// Append userA to user1's team, userB to user2's team, and userA, userB, userC to user3's team
db.Model(&users).Association("Team").Append(&userA, &userB, &[]User{userA, userB, userC})
// Replace teams for multiple users in a batch
// Reset user1's team to userA, user2's team to userB, and user3's team to userA, userB, and userC
db.Model(&users).Association("Team").Replace(&userA, &userB, &[]User{userA, userB, userC})
In GORM, the Replace, Delete, and Clear methods in Association Mode primarily affect the foreign key references, not the associated records themselves. Understanding and managing this behavior is crucial for data integrity.
UnscopedFor scenarios requiring actual deletion of associated records, the Unscoped method alters this behavior.
deleted_at field) without removing them from the database.db.Model(&user).Association("Languages").Unscoped().Clear()
// db.Unscoped().Model(&user)
db.Unscoped().Model(&user).Association("Languages").Unscoped().Clear()
Association tags in GORM are used to specify how associations between models are handled. These tags define the relationship's details, such as foreign keys, references, and constraints. Understanding these tags is essential for setting up and managing relationships effectively.
| Tag | Description |
|---|---|
foreignKey | Specifies the column name of the current model used as a foreign key in the join table. |
references | Indicates the column name in the reference table that the foreign key of the join table maps to. |
polymorphic | Defines the polymorphic type, typically the model name. |
polymorphicValue | Sets the polymorphic value, usually the table name, if not specified otherwise. |
many2many | Names the join table used in a many-to-many relationship. |
joinForeignKey | Identifies the foreign key column in the join table that maps back to the current model's table. |
joinReferences | Points to the foreign key column in the join table that links to the reference model's table. |
constraint | Specifies relational constraints like OnUpdate, OnDelete for the association. |