Как рассчитать красноречивые отношения без проблемы N + 1 и загрузки всей модели?

Я показываю список категорий и количество статей в каждой категории. Я получаю ожидаемый результат, но у меня проблема N + 1.

мой CategoriesController индексная функция:

public function index()
{
return View::make('categories.index', [
'articleCategories' => Category::where('type', 'articles')->orderBy('name')->get(),
]);
}

Category Модель имеет много отношения к статьям:

public function articles()
{
return $this->hasMany('Article');
}

мой categories.index Посмотреть:

@foreach($articleCategories as $articleCategory)
<p>
{{ HTML::link(URL::route('articles.category', array('category' => Str::slug($articleCategory->name))), $articleCategory->name) }}
{{ $articleCategory->articles->count() }}
</p>
@endforeach

Редактировать: Это работает, если я хочу загрузить все связанные статьи, но так как мне нужна только категория pr количества статей, это кажется излишним. Будет ли загружать статьи и делать ->count() повлиять на производительность? Или это лучший способ сделать это?

0

Решение

// helper relation
public function articlesCount()
{
return $this->hasOne('Article')->selectRaw('category_id, count(*) as aggregate')->groupBy('category_id');
}

// and accessor for fetching it easier
public function getArticlesCountAttribute()
{
if ( ! array_key_exists('articlesCount', $this->relations)) $this->load('articlesCount');

return $this->getRelation('articlesCount')->aggregate;
}

Тогда вы можете сделать это:

// eager load in single query
$categories = Category::with('articlesCount')->get();

// thanks to accessor this will return value and load relation only if needed
$categories->first()->articlesCount;
1

Другие решения

Попробуй это

public function index()
{
return View::make('categories.index', [
'category' => Category::with('articles')->where('type', 'articles')->orderBy('name')->get(),
]);
}

Теперь в представлении для получения данных категории просто выполните

$category->type или если у вас есть поле имени, вы можете получить имя по$category->name

Чтобы получить статьи категории вы можете сделать

foreach($category->articles as $article)
// do something with the articles
@endforeach

Чтобы получить количество статей в категории, сделайте $category->articles->count();

Убедитесь, что вы также прочитали документацию Нетерпеливая загрузка однажды это действительно очень помогает.

0

По вопросам рекламы [email protected]