gpt4 book ai didi

java - 使用 Hibernate 实体的域实体?

转载 作者:行者123 更新时间:2023-12-01 18:47:55 26 4
gpt4 key购买 nike

在我目前在 Spring Boot 应用程序中与 DDD 的斗争中,我陷入了僵局。我知道我的域实体不应该与基础设施层(Hibernate 实体所在的位置)有任何连接。当然,我的领域层依赖于 Hibernate 实体的数据来完成其操作。

因此,在我的应用程序层中,我的服务加载 Hibernate 实体并将其传递到域实体中。

这是我的域实体的示例:

package com.transportifygame.core.domain.objects;

import com.transportifygame.core.domain.OperationResult;
import com.transportifygame.core.domain.constants.Garages;
import com.transportifygame.core.domain.exceptions.garages.NotAvailableSpotException;
import com.transportifygame.core.infrastructure.entities.CompanyEntity;
import com.transportifygame.core.infrastructure.entities.GarageEntity;
import com.transportifygame.core.infrastructure.entities.LocationEntity;
import lombok.AllArgsConstructor;
import lombok.Getter;
import lombok.RequiredArgsConstructor;
import lombok.Setter;

@Getter
@Setter
@RequiredArgsConstructor
@AllArgsConstructor

public class Garage {

private GarageEntity garage;

public static Integer getAvailableSlots(Garages.Size size) {
switch (size) {
case SMALL:
return Garages.Slots.SMALL;
case MEDIUM:
return Garages.Slots.MEDIUM;
case LARGE:
return Garages.Slots.LARGE;
case HUGE:
return Garages.Slots.HUGE;
}

return 0;
}

public static Double getGaragePriceToBuy(Garages.Size size) {
switch (size) {
case SMALL:
return Garages.Prices.BUY_SMALL;
case MEDIUM:
return Garages.Prices.BUY_MEDIUM;
case LARGE:
return Garages.Prices.BUY_LARGE;
case HUGE:
return Garages.Prices.BUY_HUGE;
}

return 0.0;
}

public static OperationResult<GarageEntity, Object> buy(
Garages.Size size,
CompanyEntity company,
LocationEntity location
) {
// As we had changed the company object, we have to refresh
var newGarage = new GarageEntity();
newGarage.setCompany(company);
newGarage.setLocation(location);
newGarage.setSize(size.ordinal());
newGarage.setSlotsAvailable(Garage.getAvailableSlots(size));

return new OperationResult<>(newGarage, null);
}

public static void hasAvailableSpot(GarageEntity garage) throws NotAvailableSpotException {
if (garage.getSlotsAvailable() == 0) {
throw new NotAvailableSpotException();
}
}

public static OperationResult<GarageEntity, Object> addFreeSlot(GarageEntity garage) {
garage.setSlotsAvailable(garage.getSlotsAvailable() - 1);

return new OperationResult<>(garage, null);
}

public static OperationResult<GarageEntity, Object> removeFreeSlot(GarageEntity garage) {
garage.setSlotsAvailable(garage.getSlotsAvailable() + 1);

return new OperationResult<>(garage, null);
}

}

现在的问题是,这是向域实体提供所需数据的正确方法吗?如果不是,正确的方法是什么?

它是否使用工厂来构建基于 hibernate 实体的域实体?域实体应该镜像 hibernate 实体属性吗?

我还了解到有些人使用该方法在 Hibernate 实体中添加域实体的逻辑,但我认为这不是正确的方法。

最佳答案

我建议熟悉 DDD 存储库模式。简而言之,存储库应该模仿内存中的集合来检索和存储您的域对象。

因此,您最终可以通过两种方式使用 hibernate 设计应用程序:

  1. 编写一个高级存储库接口(interface),为您的域实体公开“查找”、“保存”等。存储库接口(interface)和返回的实体应该是“纯域逻辑”。也就是说:没有持久性或 hibernate 问题。在您的情况下,它将是 Garage 对象,其中实现了所有业务方法。在存储库实现中(可能驻留在另一个层/包中),您将处理 Hibernate 行表示(Hibernate 实体)和实体管理器以完成接口(interface)的契约(Contract)。在您的情况下,这是 Garage 对象映射到(哑)GarageEntity 并通过 EntityManagerSession 持久化的地方>.

  2. 您可以接受一些持久性/hibernate 泄漏到您的域中。这将导致有一个 Garage 域对象,其中包含同一源文件上的所有业务逻辑和 hibernate/jpa 注释(或 xml)(xml 可以是单独的)。在这种情况下,存储库可以是直接的 jpa 存储库实现。

无论如何,DDD 中的逻辑始终是相同的:

在应用程序服务(某些GarageService - 用例的入口点)中,查询存储库以获取执行业务操作的域对象。然后应用程序服务再次将修改后的实体存储在存储库中。

关于java - 使用 Hibernate 实体的域实体?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59791947/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com