0

我有以下控制器blog

public function show($slug)
{
    $post = Blog::where('slugs', '=', $slug)->first();

    $vars['pageTitle'] = Config::get('site.pageTitle') . $post['title'];

    // The breadcrumbs... needs to be repopulated every page
    $vars['breadCrumbs'] = [[
        'url'   => action('SimpleController@index'),
        'title' => 'CovertDEV'
    ],[
        'url'   => action('BlogController@index'),
        'title' => 'Blog'
    ],[
        'url'   => route('blog_post', ['slug' => $slug]),
        'title' => $post['title']
    ]];

    $vars['blog'] = $post;

    // The following line is what the question is about
    $vars['comments'] = $post->Blog_comments->groupBy('comment_id');

    return view('blog', $vars);
}

我的博客模型最初看起来像

namespace App;

use Illuminate\Database\Eloquent\Model;

class Blog extends Model
{
    //

    public function tags()
    {
        return $this->belongsToMany('App\Tag');
    }

    public function blog_comments()
    {
        return $this->hasMany('App\Blog_comment');
    }

    public function users()
    {
        return $this->belongsTo('App\User', 'user_id');
    }
}

和 blog_comments

namespace App;

use Illuminate\Database\Eloquent\Model;

class Blog_comment extends Model
{
    //

    public function users()
    {
        return $this->belongsTo('App\User', 'user_id');
    }

    public function blogs()
    {
        return $this->belongsTo('App\Blog');
    }

}

我正在检索帖子的评论并将其显示在页面上。一切都运行良好...但后来我尝试改用我的关系...(将那条线更改为...)

$vars['comments'] = $post->blog_comments()->get()->groupBy('comment_id');

这同样有效。我的帖子不大,只是一个简单的“Lorem Ipsum”测试文本和 7 条评论来测试该功能,所以我看不出这两种方法的加载时间有任何差异。

还是它们都一样?以前的方式看起来更好,但我不确定我是否按预期实施 ORM,如果没有,那么哪种方式更好?

但我是 Laravel 的新手(这是我的第一个框架),所以我不知道......哪种方式更好/更快/更高效?也许有一种我还不知道的更好/更有效的方法来实现这一点。

对不起,如果这看起来很明显。我想在继续项目之前解决这个问题……不想在找到更好的方法将信息联系在一起之后重做所有事情。

提前致谢!

我的数据库转储是

CREATE TABLE `blog_comments` (
  `id` int(10) UNSIGNED NOT NULL,
  `user_id` int(10) UNSIGNED NOT NULL,
  `blog_id` int(10) UNSIGNED NOT NULL,
  `comment_id` int(11) NOT NULL DEFAULT '0',
  `comment` text COLLATE utf8mb4_unicode_ci NOT NULL,
  `likes` int(11) NOT NULL DEFAULT '0',
  `deleted_at` timestamp NULL DEFAULT NULL,
  `created_at` timestamp NULL DEFAULT NULL,
  `updated_at` timestamp NULL DEFAULT NULL
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_unicode_ci;

--
-- Dumping data for table `blog_comments`
--

INSERT INTO `blog_comments` (`id`, `user_id`, `blog_id`, `comment_id`, `comment`, `likes`, `deleted_at`, `created_at`, `updated_at`) VALUES
(1, 1, 2, 0, 'Interesting post. Very well structured and good points throughout! One question though, Enim blandit volutpat maecenas volutpat blandit aliquam etiam erat velit?', 0, NULL, '2018-02-24 20:31:45', '2018-02-24 20:31:45'),
(2, 2, 2, 1, 'Thank you, very interesting question. Unique point of view. Semper viverra nam libero justo. Sit amet commodo nulla facilisi. Blandit massa enim nec dui nunc. Eget velit aliquet sagittis id consectetur purus ut faucibus.', 0, NULL, '2018-02-25 02:26:32', '2018-02-25 02:26:32'),
(3, 1, 2, 1, 'Indeed! I can now finally lorem ipsum by myself all the way commodo nulla facilisi!', 0, NULL, '2018-02-25 04:36:18', '2018-02-25 04:36:18'),
(4, 2, 2, 0, 'Weird, I thought I had more information posted. Must have being moderated out of existence then.', 0, NULL, '2018-02-25 09:18:58', '2018-02-25 09:18:58'),
(5, 1, 2, 4, 'Sorry about that, you had quite a few needless stuff in there. Had to tweak your post a bit. Otherwise, everything is lorem ipsum!', 0, NULL, '2018-02-25 12:53:18', '2018-02-25 12:53:18'),
(6, 2, 2, 3, 'Glad that it works for you!', 0, NULL, '2018-02-26 05:35:46', '2018-02-26 05:35:46'),
(7, 1, 2, 6, 'Thank you, feels good!', 0, NULL, '2018-02-25 07:08:33', '2018-02-25 07:08:33');

--
-- Indexes for dumped tables
--

--
-- Indexes for table `blog_comments`
--
ALTER TABLE `blog_comments`
  ADD PRIMARY KEY (`id`),
  ADD KEY `blog_comments_user_id_index` (`user_id`),
  ADD KEY `blog_comments_blog_id_index` (`blog_id`);

我需要结果数组如下所示:

Array
(
    [comment_id] => Array
        (
            [0] => Array
                (
                    [id] => 1
                    [user_id] => 1
                    [blog_id] => 2
                    [comment_id] => 0
                    [comment] => Interesting post. Very well structured and good points throughout! One question though, Enim blandit volutpat maecenas volutpat blandit aliquam etiam erat velit?
                    [likes] => 0
                    [deleted_at] => 
                    [created_at] => 2018-02-24 14:31:45
                    [updated_at] => 2018-02-24 14:31:45
                )

            [1] => Array
                (
                    [id] => 4
                    [user_id] => 2
                    [blog_id] => 2
                    [comment_id] => 0
                    [comment] => Weird, I thought I had more information posted. Must have being moderated out of existence then.
                    [likes] => 0
                    [deleted_at] => 
                    [created_at] => 2018-02-25 03:18:58
                    [updated_at] => 2018-02-25 03:18:58
                )
        )
)

comment_id是一样的parent_id...我最初命名它comment_id现在我不想改变它。

任何comment_id 为0 的评论都是父评论,任何comment_if 为任意数字的评论都是该评论的子评论。

可能引起了一些混乱。对于那个很抱歉。

我只想知道生成这种数组的最有效方法是什么。我已经经历了 3 或 4 种方法,现在我想知道我是否遗漏了什么。

4

1 回答 1

1

我认为这不是那么干净,但我相信比依赖Collection更有效的解决方案groupBy

$post = Blog::with(['blog_comments' => function($query) {
    $query->groupBy('comment_id');
}])->where('slugs', $slug)->first();

$vars['comments'] = $post->blog_comments;

尽管如果您不期望有数千条评论,那么您当前的实现是可以的。

使用它更有效,因为您将groupBy工作负载分配给数据库管理器而不是php本身(使用Collection时)groupBy

您还可以声明另一个适合您的分组需求的关系。

// Blog.php
public function groupedBlogComments()
{
    return $this->hasMany(BlogComment::class)
                ->groupBy('comment_id');

    // I think this can be `return $this->blogComments()
    //                                  ->groupBy('comment_id');
}

// you can still have this, if ever you do not want grouped comments
public function blogComments()
{
    return $this->hasMany(BlogComment::class);
}

// Controller
$vars['comments'] = $post->groupedBlogComments;

或者更好的是,在您的BlogComment模型中定义一个范围,并在任何地方使用它。

// BlogComment.php
public function scopeGroupByCommentId($query)
{
    return $query->groupBy('comment_id');
}

// You can eager load your relationships and use the scope.
$post = Blog::with(['blog_comments' => function($query) {
    $query->groupByCommentId();
}])->where('slugs', $slug)->first();

$vars['comments'] = $post->blog_comments;

如果有任何错误,请不要犹豫,纠正我。

我希望我明白你的意思。

于 2018-02-27T02:13:01.717 回答