Laravel parent / child 同一模特的关系
Laravel parent / child relationship on the same model
设置和虚拟数据
我有一个名为 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 中映射这种关系,以便我可以查询模型并获取它的 parent(它总是有一个 parent),并且 child 类别(如果存在)。
我在类别模型中使用以下方法定义了关系:
public function children()
{
return $this->hasMany('Category', 'parent_id', 'cat_id');
}
public function parent()
{
return $this->belongsTo('Category', 'parent_id');
}
问题
我可以让 parent 名称正常工作,使用
$category = Category::findOrFail($id);
return $category->parent->cat_name;
但是,我不明白如何获得 child objects。
我试过:
$category = Category::findOrFail($id);
$children = $category->children();
但是当我 dd($children) 它没有输出我所期望的。
调用关系函数 (->children()
) 将 return 关系的一个实例 class。您要么需要调用 get()
,要么只使用 属性:
$children = $category->children()->get();
// or
$children = $category->children;
进一步说明
实际上 children()
和 children
是完全不同的东西。 children()
只是调用您为关系定义的方法。方法 return 是 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
中首先是一些代码,用于检查 "normal" 属性,然后是 return。最后,在该方法的末尾,如果定义了关系方法,则调用 getRelationshipFromMethod
。
然后它将检索关系的结果,return。
在模型中设置并尝试:
public function children()
{
return $this->hasMany(self::class, 'parent_id');
}
public function grandchildren()
{
return $this->children()->with('grandchildren');
}
设置和虚拟数据
我有一个名为 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 中映射这种关系,以便我可以查询模型并获取它的 parent(它总是有一个 parent),并且 child 类别(如果存在)。
我在类别模型中使用以下方法定义了关系:
public function children()
{
return $this->hasMany('Category', 'parent_id', 'cat_id');
}
public function parent()
{
return $this->belongsTo('Category', 'parent_id');
}
问题
我可以让 parent 名称正常工作,使用
$category = Category::findOrFail($id);
return $category->parent->cat_name;
但是,我不明白如何获得 child objects。
我试过:
$category = Category::findOrFail($id);
$children = $category->children();
但是当我 dd($children) 它没有输出我所期望的。
调用关系函数 (->children()
) 将 return 关系的一个实例 class。您要么需要调用 get()
,要么只使用 属性:
$children = $category->children()->get();
// or
$children = $category->children;
进一步说明
实际上 children()
和 children
是完全不同的东西。 children()
只是调用您为关系定义的方法。方法 return 是 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
中首先是一些代码,用于检查 "normal" 属性,然后是 return。最后,在该方法的末尾,如果定义了关系方法,则调用 getRelationshipFromMethod
。
然后它将检索关系的结果,return。
在模型中设置并尝试:
public function children()
{
return $this->hasMany(self::class, 'parent_id');
}
public function grandchildren()
{
return $this->children()->with('grandchildren');
}