如何在Laravel框架中使用Eloquent或Fluent选择随机行?

我知道,通过使用SQL,您可以按RAND()进行排序。但是,我希望获得随机行,而不是在初始查询之前对记录的数量进行计数。

什么好主意吗?


当前回答

tl;dr:它现在已经被实现到Laravel中了,请看下面的“编辑3”。


遗憾的是,到今天为止,->orderBy(DB::raw('RAND()'))提出的解决方案有一些警告:

它不是数据库不可知论者。例如,SQLite和PostgreSQL使用RANDOM() 更糟糕的是,这个解决方案不再适用,因为这个改变: $direction = strtolower($direction) == 'asc' ?'asc': 'desc';

现在你可以使用orderByRaw()方法:->orderByRaw('RAND()')。然而,这仍然不是数据库不可知的。

CodeIgniter实现了一个特殊的RANDOM排序方向,在构建查询时用正确的语法替换。而且它似乎很容易实现。看起来我们有一个改善Laravel的候选人:)

更新:这是关于GitHub上的问题,和我悬而未决的拉请求。

编辑2:让我们开门见山吧。从Laravel 5.1.18开始,你可以在查询生成器中添加宏:

use Illuminate\Database\Query\Builder;

Builder::macro('orderByRandom', function () {

    $randomFunctions = [
        'mysql'  => 'RAND()',
        'pgsql'  => 'RANDOM()',
        'sqlite' => 'RANDOM()',
        'sqlsrv' => 'NEWID()',
    ];

    $driver = $this->getConnection()->getDriverName();

    return $this->orderByRaw($randomFunctions[$driver]);
});

用法:

User::where('active', 1)->orderByRandom()->limit(10)->get();

DB::table('users')->where('active', 1)->orderByRandom()->limit(10)->get();

编辑3:终于!从Laravel 5.2.33 (changelog, PR #13642)开始,你可以在randomorder()中使用本机方法:

User::where('active', 1)->inRandomOrder()->limit(10)->get();

DB::table('users')->where('active', 1)->inRandomOrder()->limit(10)->get();

其他回答

这很好,

$model=Model::all()->random(1)->first();

你也可以改变随机函数中的参数来获得多条记录。

注意:如果你有大量的数据,不建议这样做,因为这会先获取所有行,然后返回随机值。

我宁愿先说明,否则就失败:

$collection = YourModelName::inRandomOrder()
  ->firstOrFail();

在你的模型中添加这个:

public function scopeRandomize($query, $limit = 3, $exclude = [])
{
    $query = $query->whereRaw('RAND()<(SELECT ((?/COUNT(*))*10) FROM `products`)', [$limit])->orderByRaw('RAND()')->limit($limit);
    if (!empty($exclude)) {
        $query = $query->whereNotIn('id', $exclude);
    }
    return $query;
}

然后是route/controller

$data = YourModel::randomize(8)->get();

Laravel有一个内置的方法来打乱结果的顺序。

以下是文档中的一段话:

shuffle()

shuffle方法随机洗牌集合中的项:

$collection = collect([1, 2, 3, 4, 5]);

$shuffled = $collection->shuffle();

$shuffled->all();

// [3, 2, 5, 1, 4] - (generated randomly)

您可以在这里看到文档。

您可以轻松地使用此命令:

//问题:模型的名称 //从DB中抽取10行

$questions = Question::orderByRaw('RAND()')->take(10)->get();