【问题标题】:How to have multiple same polymorphic relations in one model in PHP-Laravel?如何在 PHP-Laravel 的一个模型中拥有多个相同的多态关系?
【发布时间】:2026-01-29 06:35:01
【问题描述】:

我正在处理一个挑战,我想创建一个多态模型本地化。这将包括长文本列,如 defaulten_usde_de、...

现在想象另一个模型产品。产品通常具有文本属性,如名称、描述、... 下面是我可以使用 Localizations 的多态关系的部分。 localizations 表应该具有这种类型的结构:

id localizable_id localizable_type default en_us de_de
1 25 App\Model\Product Phone null Telefon
2 25 App\Model\Product The best phone on the market without an audio jack null Das beste Telefon auf dem Markt ohne Audioanschluss
3 15 App\Model\Article Top 10 products null Top 10 Produkte
4 15 App\Model\Job Salesman null Verkäufer

如果我希望 Product 具有可本地化的名称和描述,那么根据 Laravel 文档,您应该期望在我的代码中出现这样的内容:

class Product extends Model
{
  public function name() // Expecting model with default attribute 'Phone'
  {
    return $this->morphOne(Localization::class,'localizable');
  }

  public function description() // Expecting model with default attribute 'The best phone on the market without an audio jack'
  {
    return $this->morphOne(Localization::class,'localizable');
  }
}

无论多么明显,它都无法正常工作,因为我不能指望两个相同的方法返回不同的值。

另一方面,如果我想遵循 Laravel 的约定,Localization 模型应该是这样的:

class Localizable extends Model
{
  public function localizable()
  {
    return $this->morphTo(__FUNCTION__,'localizable_type','localizable_id');
  }
}

正如您在上面的示例中看到的,多态关系与我所需要的相反。问题是我想要一张包含所有字符串的表,这些字符串可能会被翻译(localizables)并在许多其他模型中使用它们,而不仅仅是 Product,例如 商店工作文章、...

我需要以某种方式实现区分 localizable_types 以及它应该与哪个列/属性相关。实现这一目标的最佳方法是什么?

【问题讨论】:

  • 为什么不使用 Laravel 内置的本地化支持?如果您希望能够通过 Web 界面编辑翻译文件,可以使用这样的软件包。 github.com/barryvdh/laravel-translation-manager
  • @miken32 首先我需要能够为翻译机构创建特定的包(通过 laravel 的本地化是不可能的)。其次,我需要将应用本地化与数据本地化分开。第三,它一般不能回答问题 - 另一个示例可能适用于图像,其中一个用户可以拥有个人资料图像和背景图像(具有用于例如文章的多态图像)
  • 不会简单地将另一列添加到具有类型(名称、描述等)的表中吗?然后根据它过滤关系?
  • @miken32 我想到了可以用于检索现有记录的列本地化.localizable_context,但我必须以某种方式修改本地化的 save() 方法来添加这个关系名称。而且我不知道如何从 save 方法访问这些信息

标签: php laravel polymorphism relation


【解决方案1】:

我终于找到了满足我期望的解决方法。即使它不是最漂亮的(或Laravel)方式。我从在 GitHub here 上找到的类似斗争中得到启发。

此解决方案的重点是覆盖 ProductgetMorphClass() 方法,该方法用于确定 *_type 列值。我的产品型号:

class Product extends Model
{
  protected $morphClass = null; // Create an attribute that will not be saved into the DB
  public function getMorphClass() // Method for determinating the type value
  {
    return $this->morphClass? : self::class;
  }

  public function name()
  {
    $this->morphClass = self::class . '.name'; // App\Models\Product.name
    return $this->morphOne(Localization::class, 'localizable');
  }

  public function description()
  {
    $this->morphClass = self::class . '.description'; // App\Models\Product.description
    return $this->morphOne(Localization::class, 'description');
  }

}

上面的修改会影响 Laravel 如何将 Localization 保存到数据库中。现在,需要从关系的另一端完成一些工作 - custom polymorphic types。更新 app/Providers/AppServiceProvider.php 文件中的 AppServiceProvider 类。您必须将 morphMap 添加到 boot() 方法中:

/**
 * Bootstrap any application services.
 *
 * @return void
 */
public function boot()
{
  // ...any of your previous modifications...

  Relation::morphMap([
    Product::class . '.name' => Product::class,
    Product::class . '.description' => Product::class,
  ]);
}

但请注意:一旦您决定在模型中使用多个相同类型的变形,您必须在调用 morphOne() 或 @987654332 之前更改每个动态方法中的受保护属性 $morphClass @。否则,您可能会将描述保存到名称中,反之亦然。

另外,当你在另一边使用morphTo() 方法时,你只会得到Product - 如果你需要知道Product 上的什么属性> 正在关联模型,我建议您创建一些提取上下文的方法。

请随时评论此方法中的任何潜在威胁。

【讨论】:

    最近更新 更多