Nibernato Unidirecional One-One Mapping Problem
-
22-09-2019 - |
Pergunta
Estou tentando criar um relacionamento individual unidirecional usando o Nibernate.
Exemplo: um pedido é dado por um cliente.
Customer{ID, Name, Address}
OrderN{ID, Customer, OrderDate}
Aqui, o ordern.customer-field pretende armazenar o cliente.id como um FK. E esse campo não tem nenhuma restrição única.
(A tabela Ordern recebe esse nome para evitar conflitos de palavras-chave SQL.)
O problema é que, depois de executar esse código C#, o Ordern.Customer-Field está armazenando um valor nulo.
Mas deveria armazenar o ID do cliente. Ou seja, 1.
E se eu acrescentar <property name="Customer" column="Customer" />
Em ordem.hbm.xml, uma exceção é lançada:
Could not determine type for: NHibernate__One_To_One__Order_Customer.BO.Customer, NHibernate__One_To_One__Order_Customer.BO, Version=1.0.0.0, Culture=neutral, PublicKeyToken=null, for columns: NHibernate.Mapping.Column(Customer)
Como resolver este problema?
Pode ser que não seja um relacionamento individual. Na verdade, estou tentando entender como está o <one-to-one />
tag usado. Alguém pode me ajudar a esse respeito?
Customer.sql
CREATE TABLE [dbo].[Customer](
[ID] [int] IDENTITY(1,1) NOT NULL,
[Name] [varchar](50) NULL,
[Address] [varchar](50) NULL,
CONSTRAINT [PK_Customer] PRIMARY KEY CLUSTERED
(
[ID] ASC
)WITH (PAD_INDEX = OFF, STATISTICS_NORECOMPUTE = OFF, IGNORE_DUP_KEY = OFF, ALLOW_ROW_LOCKS = ON, ALLOW_PAGE_LOCKS = ON) ON [PRIMARY]
) ON [PRIMARY]
Customer.cs
public class Customer
{
private int _id;
public virtual int ID
{
get { return _id; }
set { _id = value; }
}
private string _name;
public virtual string Name
{
get { return _name; }
set { _name = value; }
}
private string _address;
public virtual string Address
{
get { return _address; }
set { _address = value; }
}
}
Customer.hbm.xml
<?xml version="1.0" encoding="utf-8" ?>
<hibernate-mapping
xmlns="urn:nhibernate-mapping-2.2"
>
<class name="NHibernate__One_To_One__Order_Customer.BO.Customer, NHibernate__One_To_One__Order_Customer.BO" table="Customer">
<id name="ID" >
<generator class="native" />
</id>
<property name="Name" column="Name" />
<property name="Address" column="Address" />
</class>
</hibernate-mapping>
Ordern.sql
CREATE TABLE [dbo].[OrderN](
[ID] [int] IDENTITY(1,1) NOT NULL,
[Customer] [int] NULL,
[OrderDate] [datetime] NULL,
CONSTRAINT [PK_Order] PRIMARY KEY CLUSTERED
(
[ID] ASC
)WITH (PAD_INDEX = OFF, STATISTICS_NORECOMPUTE = OFF, IGNORE_DUP_KEY = OFF, ALLOW_ROW_LOCKS = ON, ALLOW_PAGE_LOCKS = ON) ON [PRIMARY]
) ON [PRIMARY]
GO
ALTER TABLE [dbo].[OrderN] WITH CHECK ADD CONSTRAINT [FK_Order_Customer] FOREIGN KEY([Customer])
REFERENCES [dbo].[Customer] ([ID])
GO
ALTER TABLE [dbo].[OrderN] CHECK CONSTRAINT [FK_Order_Customer]
Ordern.cs
public class OrderN
{
private int _id;
public virtual int ID
{
get { return _id; }
set { _id = value; }
}
private Customer _customer;
public virtual Customer Customer
{
get { return _customer; }
set { _customer = value; }
}
private DateTime _orderDate;
public virtual DateTime OrderDate
{
get { return _orderDate; }
set { _orderDate = value; }
}
}
Ordern.hbm.xml
<?xml version="1.0" encoding="utf-8" ?>
<hibernate-mapping
xmlns="urn:nhibernate-mapping-2.2"
>
<class name="NHibernate__One_To_One__Order_Customer.BO.OrderN, NHibernate__One_To_One__Order_Customer.BO" table="OrderN">
<id name="ID">
<generator class="native" />
</id>
<property name="OrderDate" column="OrderDate"/>
<one-to-one
name="Customer"
class="NHibernate__One_To_One__Order_Customer.BO.Customer, NHibernate__One_To_One__Order_Customer.BO" />
</class>
</hibernate-mapping>
Programa principal
OrderN o = new OrderN();
o.OrderDate = DateTime.Now;
o.Customer = new Repository<Customer>().Get<Customer>(1);
Repository<OrderN> rep = new Repository<OrderN>();
rep.Save(o);
Solução
Então, um cliente só pode ter um pedido? Espero que seja grande!
Este não é um relacionamento individual, é um para muitos. O cliente é o lado, e o pedido é de muitos (o cliente tem pedidos). Tente mapear assim:
<?xml version="1.0" encoding="utf-8" ?>
<hibernate-mapping
xmlns="urn:nhibernate-mapping-2.2"
>
<class name="NHibernate__One_To_One__Order_Customer.BO.OrderN, NHibernate__One_To_One__Order_Customer.BO" table="OrderN">
<id name="ID">
<generator class="native" />
</id>
<property name="OrderDate" column="OrderDate"/>
<many-to-one
name="Customer"
class="NHibernate__One_To_One__Order_Customer.BO.Customer, NHibernate__One_To_One__Order_Customer.BO"
column="Customer" />
</class>
</hibernate-mapping>