What is the preferred method for creating children objects?
- Adding the child to a collection on it's parent and calling update on the parent, or
- Creating the child explicitly, add the child to a collection on the parent, and updating the parent?
I'm struggling with #1. And it's hard enough that I'm thinking #2 is preferable (less "magic").
Not sure what would be the "preferred method" (it could depend on who does prefer). But I can share my way
If the parent is a root entity, then we should save all the reference tree in in one shot: session.SaveOrUpdate(parent)
.
E.g. Employee
has collection of Educations
. In such case, it should go like this
- create
Education
give it reference to employee,
Add()
it to collection employee.Educations
.
- Have/make the mapping inverse, cascade="all-delete-orphan" ...
- NHibernate will do what we expect on session.SaveOrUpdate(parent)
Some snippets in xml:
<class name="Employee" table="..." lazy="true"
optimistic-lock="version" dynamic-update="true" batch-size="25" >
<cache usage="read-write" region="ShortTerm"/>
<id name="ID" column="Employee_ID" generator="native" />
<bag name="Educations" lazy="true" inverse="true"
batch-size="25" cascade="all-delete-orphan" >
<key column="Employee_ID" />
<one-to-many class="Education" />
</bag>
...
And the Education
<class name="Education" table="..." lazy="true" batch-size="25>
<cache usage="read-write" region="ShortTerm"/>
<id name="ID" column="Education_ID" generator="native" />
<many-to-one not-null="true" name="Employee" class="Employee"
column="Employee_ID" />
...
in fluent:
public class EmployeeMap : ClassMap<Employee>
{
public EmployeeMap()
{
BatchSize(25)
...
HasMany(x => x.Educations)
.AsBag()
.BatchSize(25)
.LazyLoad()
.Cascade.AllDeleteOrphan()
.Inverse()
.KeyColumn("Employee_ID")
public class EducationMap : ClassMap<Education>
{
public EducationMap()
{
...
References(x => x.Employee, "Employee_ID")
.Not.Nullable()
...
And now the C# relations:
// business
var employee = ...;
var education = new Education
{
Employee = employee,
...
};
employee.Educations.Add(education);
// DAO
session.SaveOrUpdate(employee);
If the parent is not root, there is just a relation (Employee
has collection of Subordinates
of type Employee
), keep the persisting separated