在網上商城上,我們經常可以看到多級分類、子分類、甚至無限極分類。本文將向你展示如何優雅的通過 Laravel Eloquent 將其實現。
我們會創建一個微型項目來展示兒童商店的分類,總共有 5 級,如下:

數據庫遷移
簡單的數據表結構:
Schema::create('categories', function (Blueprint $table) {
$table->bigIncrements('id');
$table->string('name');
$table->unsignedBigInteger('category_id')->nullable();
$table->foreign('category_id')->references('id')->on('categories');
$table->timestamps();
});
只有一個 name 字段, 關聯到其自身。所以,大部分父級分類 category_id = NULL,每一個子分類都有一個 parent_id
數據表數據如下:

Eloquent 模型和關聯關系
首先,在 app/Category.php 創建一個簡單的 hasMany() 方法, 分類可能擁有其自分類:
class Category extends Model
{
public function categories()
{
return $this->hasMany(Category::class);
}
}
好戲開場 本文最妙 “計策”。你知道可以向這樣描述 遞歸 關系嗎?如下:
public function childrenCategories()
{
return $this->hasMany(Category::class)->with('categories');
}
因此,如果調用 Category::with(‘categories’),將得到下級 “子分類”,但是通過 Category::with(‘childrenCategories’) 將能幫你實現無限極。
路由和控制器方法
現在,讓我們嘗試顯示所有類別和子類別,如上例所示。
在 routes/web.php,我們添加以下內容:
Route::get('categories', 'CategoryController@index');
app/Http/CategoryController.php 如下所示:
public function index()
{
$categories = Category::whereNull('category_id')
->with('childrenCategories')
->get();
return view('categories', compact('categories'));
}
我們僅加載父類別,將子類別作為關系。簡單吧?
視圖和遞歸子視圖
最后,渲染到頁面。 在 resources/views/categories.blade.php 文件:
<ul>
@foreach ($categories as $category)
<li>{{ $category->name }}</li>
<ul>
@foreach ($category->childrenCategories as $childCategory)
@include('child_category', ['child_category' => $childCategory])
@endforeach
</ul>
@endforeach
</ul>
我們先遍歷了最頂級的父類別,然后遍歷出父類的子類別,然后使用 @include 加載子類別的子類別......
最好的部分是 resources/views/admin/child_category.blade.php 將使用遞歸加載自身。看代碼:
<li>{{ $child_category->name }}</li>
@if ($child_category->categories)
<ul>
@foreach ($child_category->categories as $childCategory)
@include('child_category', ['child_category' => $childCategory])
@endforeach
</ul>
@endif
在 child_category.blade.php 內部,我們包含了 @include(‘child_category’),因此只要當前子類別中有類別,模板就會遞歸地加載子類別。
就是這樣!我們擁有無限級別的子類別 - 無論是在數據庫還是關聯關系或是視圖中
更多學習內容請訪問:
騰訊T3-T4標准精品PHP架構師教程目錄大全,只要你看完保證薪資上升一個台階(持續更新)

