php - Laravel在同一模型上的父/子关系

标签 php laravel

设置和虚拟数据

我有一个名为Category的简单模型,该模型具有以下架构:

|----------------------------------------------|
| cat_id   | cat_name    | parent_id           |
|----------------------------------------------|
|   1      | Home        |   0                 |
|----------------------------------------------|
|   2      | Products    |   1                 |
|----------------------------------------------| 
|   3      | Services    |   1                 |
|----------------------------------------------|
|   4      | Product A   |   2                 |
|----------------------------------------------|
|   5      | Product B   |   2                 |
|----------------------------------------------|

所需的输出

因此,您可以看到我们将得到一个非常简单的层次结构,如下所示:
Home
  - Products
      - Product A
      - Product B
  - Services

问题

我正在尝试在Laravel 4.2中映射此关系,以便我可以查询模型并获取其父级(它将始终有一个父级)以及子类别(如果存在)。

我已经使用以下方法在“类别”模型中定义了关系:
public function children()
{
    return $this->hasMany('Category', 'parent_id', 'cat_id');
}
public function parent()
{
    return $this->belongsTo('Category', 'parent_id');
}

问题

我可以使用
$category = Category::findOrFail($id);
return $category->parent->cat_name;

但是,我不知道如何获取子对象。

我试过了:
$category = Category::findOrFail($id);
$children = $category->children();

但是当我dd($ children)时,它没有输出我所期望的。

最佳答案

调用关系函数(->children())将返回关系类的实例。您或者需要先调用get()或只使用属性:

$children = $category->children()->get();
// or
$children = $category->children;

进一步说明

实际上children()children是完全不同的东西。 children()只是调用您为关系定义的方法。该方法返回一个HasMany对象。您可以使用它来应用更多的查询方法。例如:
$category->children()->orderBy('firstname')->get();

现在访问属性 children的工作方式有所不同。您从未定义过它,因此Laravel在后台做了一些魔术。

让我们看看Illuminate\Database\Eloquent\Model:
public function __get($key)
{
    return $this->getAttribute($key);
}

当您尝试访问实际上不存在的PHP对象上的属性时,将调用__get函数。
public function getAttribute($key)
{
    $inAttributes = array_key_exists($key, $this->attributes);

    // If the key references an attribute, we can just go ahead and return the
    // plain attribute value from the model. This allows every attribute to
    // be dynamically accessed through the _get method without accessors.
    if ($inAttributes || $this->hasGetMutator($key))
    {
        return $this->getAttributeValue($key);
    }

    // If the key already exists in the relationships array, it just means the
    // relationship has already been loaded, so we'll just return it out of
    // here because there is no need to query within the relations twice.
    if (array_key_exists($key, $this->relations))
    {
        return $this->relations[$key];
    }

    // If the "attribute" exists as a method on the model, we will just assume
    // it is a relationship and will load and return results from the query
    // and hydrate the relationship's value on the "relationships" array.
    $camelKey = camel_case($key);

    if (method_exists($this, $camelKey))
    {
        return $this->getRelationshipFromMethod($key, $camelKey);
    }
}

然后在getAttribute中首先是一些代码,这些代码检查“正常”属性,然后返回。最后,在该方法的最后,如果有关联方法,则将调用getRelationshipFromMethod

然后它将检索关系的结果并将其返回。

关于php - Laravel在同一模型上的父/子关系,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27816738/

相关文章:

php - Laravel/PHP/MySQL - 每小时与成千上万的用户一起存储/记录数据的最佳方式?

php - 在 laravel eloquent 中使用 time_to_sec()

php - SQL关键字搜索

php - 数据库中的特殊字符显示不正确

php - 导出 CSV |奇怪的行为

java - 如何从服务器获取响应,例如 "Upload Successful"

Laravel Vue SPA 使用 Sanctum 响应未经授权

php - 如何使用 MySQL Left Join 根据另一个表行的 id 获取结果?

php - 页面加载时自动点击 iframe 中的按钮

php - 如何在 Laravel 5 中获取上传文件的公共(public) URL