From da8e1d01911b14bf40041dc455f7d01611aa5ef9 Mon Sep 17 00:00:00 2001 From: huang <1724659546@qq.com> Date: Thu, 27 Nov 2025 20:03:14 +0800 Subject: [PATCH] =?UTF-8?q?=E4=BC=98=E5=8C=96=E7=AE=97=E6=B3=95?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../domain/recipe/recipe_generate_manager.go | 12 ++--- internal/domain/recipe/recipe_service.go | 50 ++++++++++++++++--- internal/infra/models/recipe.go | 22 ++++++++ 3 files changed, 67 insertions(+), 17 deletions(-) diff --git a/internal/domain/recipe/recipe_generate_manager.go b/internal/domain/recipe/recipe_generate_manager.go index 6296cea..cb833f8 100644 --- a/internal/domain/recipe/recipe_generate_manager.go +++ b/internal/domain/recipe/recipe_generate_manager.go @@ -266,7 +266,7 @@ func (r *recipeGenerateManagerImpl) GenerateRecipe(ctx context.Context, pigType // --------------------------------------------------------- // lp.Simplex 求解: minimize c^T * x subject to A * x = b, x >= 0 - optVal, x, err := lp.Simplex(c, A, b, 1e-8, nil) + _, x, err := lp.Simplex(c, A, b, 1e-8, nil) if err != nil { if errors.Is(err, lp.ErrInfeasible) { @@ -291,8 +291,8 @@ func (r *recipeGenerateManagerImpl) GenerateRecipe(ctx context.Context, pigType } recipe := &models.Recipe{ - Name: fmt.Sprintf("%s-%s - 自动计算配方", pigType.Breed.Name, pigType.AgeStage.Name), - Description: fmt.Sprintf("基于 %d 种原料计算的最优成本配方。计算时预估成本: %.2f元/kg", actualMaterialCount, optVal), + Name: fmt.Sprintf("%s-%s - 自动计算配方", pigType.Breed.Name, pigType.AgeStage.Name), // 提供一个默认的名称 + Description: fmt.Sprintf("基于 %d 种原料计算的最优成本配方。", actualMaterialCount), // 提供一个默认的描述 RecipeIngredients: []models.RecipeIngredient{}, } @@ -327,12 +327,6 @@ func (r *recipeGenerateManagerImpl) GenerateRecipe(ctx context.Context, pigType if totalPercentage > 1.0+1e-3 { return nil, fmt.Errorf("计算结果异常:实际原料总量超过 100%% (计算值: %.2f),请检查算法或数据配置", totalPercentage) } - // 如果 totalPercentage 小于 1.0,说明填充料被使用,这是符合预期的。 - // 此时需要在描述中说明需要添加的廉价填充料的百分比。 - if totalPercentage < 1.0-1e-4 { // 允许微小的浮点误差 - fillerPercentage := (1.0 - totalPercentage) * 100.0 - recipe.Description = fmt.Sprintf("%s。注意:配方中实际原料占比 %.2f%%,需额外补充 %.2f%% 廉价填充料", recipe.Description, totalPercentage*100.0, fillerPercentage) - } return recipe, nil } diff --git a/internal/domain/recipe/recipe_service.go b/internal/domain/recipe/recipe_service.go index d8db69b..f1a39b3 100644 --- a/internal/domain/recipe/recipe_service.go +++ b/internal/domain/recipe/recipe_service.go @@ -4,6 +4,7 @@ import ( "context" "fmt" + "git.huangwc.com/pig/pig-farm-controller/internal/infra/logs" "git.huangwc.com/pig/pig-farm-controller/internal/infra/models" "git.huangwc.com/pig/pig-farm-controller/internal/infra/repository" ) @@ -60,8 +61,10 @@ func NewRecipeService( // pigTypeID: 目标猪类型的ID。 // 返回: 生成的配方对象指针和可能的错误。 func (r *recipeServiceImpl) GenerateRecipeWithAllRawMaterials(ctx context.Context, pigTypeID uint32) (*models.Recipe, error) { + serviceCtx, logger := logs.Trace(ctx, r.ctx, "GenerateRecipeWithAllRawMaterials") + // 1. 获取猪只类型信息,确保包含了营养需求 - pigType, err := r.GetPigTypeByID(ctx, pigTypeID) + pigType, err := r.GetPigTypeByID(serviceCtx, pigTypeID) if err != nil { return nil, fmt.Errorf("获取猪类型信息失败: %w", err) } @@ -69,23 +72,54 @@ func (r *recipeServiceImpl) GenerateRecipeWithAllRawMaterials(ctx context.Contex // 2. 获取所有原料 // 我们通过传递一个非常大的 pageSize 来获取所有原料,这在大多数情况下是可行的。 // 对于超大规模系统,可能需要考虑分页迭代,但目前这是一个简单有效的策略。 - materials, _, err := r.ListRawMaterials(ctx, repository.RawMaterialListOptions{}, 1, 9999) + materials, _, err := r.ListRawMaterials(serviceCtx, repository.RawMaterialListOptions{}, 1, 9999) if err != nil { return nil, fmt.Errorf("获取所有原料列表失败: %w", err) } // 3. 调用生成器生成配方 - recipe, err := r.GenerateRecipe(ctx, *pigType, materials) + recipe, err := r.GenerateRecipe(serviceCtx, *pigType, materials) if err != nil { return nil, fmt.Errorf("生成配方失败: %w", err) } - // 4. 保存新生成的配方到数据库 - // CreateRecipe 会处理配方及其成分的保存 - if recipe, err = r.CreateRecipe(ctx, recipe); err != nil { - return nil, fmt.Errorf("保存生成的配方失败: %w", err) + // 4. 丰富配方描述:计算并添加参考价格信息 + + // 填充 RecipeIngredients 中的 RawMaterial 字段,以便后续计算成本 + rawMaterialMap := make(map[uint32]models.RawMaterial) + for _, mat := range materials { + rawMaterialMap[mat.ID] = mat + } + for i := range recipe.RecipeIngredients { + if rawMat, ok := rawMaterialMap[recipe.RecipeIngredients[i].RawMaterialID]; ok { + recipe.RecipeIngredients[i].RawMaterial = rawMat + } else { + // 理论上 GenerateRecipe 应该只使用传入的 materials 中的 RawMaterialID + // 如果出现此情况,说明 GenerateRecipe 生成了不在当前 materials 列表中的 RawMaterialID + // 这可能是一个数据不一致或逻辑错误,记录警告以便排查 + logger.Warnf("未找到 RecipeIngredient (RawMaterialID: %d) 对应的 RawMaterial,成本计算可能不准确", recipe.RecipeIngredients[i].RawMaterialID) + } } - // 5. 返回创建的配方 (现在它应该已经有了ID) + referencePrice := recipe.CalculateReferencePricePerKilogram() / 100 + recipe.Description = fmt.Sprintf("%s 计算时预估成本: %.2f元/kg。", recipe.Description, referencePrice) + + // 如果 totalPercentage 小于 100%,说明填充料被使用,这是符合预期的。 + // 此时需要在描述中说明需要添加的廉价填充料的百分比。 + totalPercentage := recipe.CalculateTotalRawMaterialProportion() + if totalPercentage < 99.99 { // 允许微小的浮点误差 + fillerPercentage := 100 - totalPercentage + recipe.Description = fmt.Sprintf("%s 注意:配方中实际原料占比 %.2f%%,需额外补充 %.2f%% 廉价填充料", recipe.Description, totalPercentage, fillerPercentage) + + } + + // 5. 保存新生成的配方到数据库 + // CreateRecipe 会处理配方及其成分的保存 + if recipe, err = r.CreateRecipe(serviceCtx, recipe); err != nil { + return nil, fmt.Errorf("保存生成的配方失败: %w", err) + } + logger.Infof("成功生成配方: %+v", recipe) + + // 6. 返回创建的配方 (现在它应该已经有了ID) return recipe, nil } diff --git a/internal/infra/models/recipe.go b/internal/infra/models/recipe.go index a5278b0..4602af1 100644 --- a/internal/infra/models/recipe.go +++ b/internal/infra/models/recipe.go @@ -13,6 +13,28 @@ func (Recipe) TableName() string { return "recipes" } +// CalculateTotalRawMaterialProportion 计算配方中所有原料的总比例 +func (r Recipe) CalculateTotalRawMaterialProportion() float32 { + var totalPercentage float32 + for _, ingredient := range r.RecipeIngredients { + totalPercentage += ingredient.Percentage + } + return totalPercentage +} + +// CalculateReferencePricePerKilogram 根据原料参考价计算配方每公斤的成本 +func (r Recipe) CalculateReferencePricePerKilogram() float32 { + var totalCost float32 + for _, ingredient := range r.RecipeIngredients { + // 确保 RawMaterial 已经被加载 + if ingredient.RawMaterial.ID == 0 { + return 0.0 + } + totalCost += ingredient.RawMaterial.ReferencePrice * ingredient.Percentage + } + return totalCost +} + // RecipeIngredient 配方原料组成模型 type RecipeIngredient struct { Model