如何在 Laravel 框架中使用 Eloquent 或 Fluent 选择随机行?
我知道通过使用 SQL,您可以通过 RAND() 进行排序。但是,我想在不计算初始查询之前的记录数的情况下获得随机行。
有任何想法吗?
Laravel >= 5.2:
User::inRandomOrder()->get();
或获取特定数量的记录
// 5 indicates the number of records
User::inRandomOrder()->limit(5)->get();
// get one random record
User::inRandomOrder()->first();
或对集合使用随机方法:
User::all()->random();
User::all()->random(10); // The amount of items you wish to receive
Laravel 4.2.7 - 5.1:
User::orderByRaw("RAND()")->get();
Laravel 4.0 - 4.2.6:
User::orderBy(DB::raw('RAND()'))->get();
拉拉维尔 3:
User::order_by(DB::raw('RAND()'))->get();
查看这篇关于 MySQL 随机行的文章。Laravel 5.2 支持这一点,对于旧版本,没有比使用RAW Queries更好的解决方案。
编辑 1:正如 Double Gras 所提到的, orderBy()自此更改以来不允许除 ASC 或 DESC 之外的任何其他内容。我相应地更新了我的答案。
编辑 2: Laravel 5.2 最终为此实现了一个包装函数。它被称为inRandomOrder()。
这工作得很好,
$model=Model::all()->random(1)->first();
您还可以更改随机函数中的参数以获取多条记录。
注意:如果您有大量数据,则不建议这样做,因为这将首先获取所有行然后返回随机值。
tl;dr:它现在已在 Laravel 中实现,请参见下面的“编辑 3”。
可悲的是,截至今天,建议的解决方案有一些警告->orderBy(DB::raw('RAND()'))
:
RANDOM()
更糟糕的是,由于此更改,此解决方案不再适用:
$direction = strtolower($direction) == 'asc' ? 'asc' : 'desc';
编辑:现在您可以使用orderByRaw()方法:->orderByRaw('RAND()')
。然而,这仍然不是 DB 无关的。
FWIW,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 ) 开始,你可以使用原生方法inRandomOrder()
:
User::where('active', 1)->inRandomOrder()->limit(10)->get();
DB::table('users')->where('active', 1)->inRandomOrder()->limit(10)->get();
您可以使用:
ModelName::inRandomOrder()->first();
在Laravel 4 和 5中,order_by
替换为orderBy
所以,它应该是:
User::orderBy(DB::raw('RAND()'))->get();
这很简单,只需检查您的 laravel 版本
Laravel >= 5.2:
User::inRandomOrder()->get();
//or to get the specific number of records
// 5 indicates the number of records
User::inRandomOrder()->limit(5)->get();
// get one random record
User::inRandomOrder()->first();
或对集合使用随机方法:
User::all()->random();
User::all()->random(10); // The amount of items you wish to receive
Laravel 4.2.7 - 5.1:
User::orderByRaw("RAND()")->get();
Laravel 4.0 - 4.2.6:
User::orderBy(DB::raw('RAND()'))->get();
拉拉维尔 3:
User::order_by(DB::raw('RAND()'))->get();
对于 Laravel 5.2 >=
使用雄辩的方法:
inRandomOrder()
inRandomOrder 方法可用于对查询结果进行随机排序。例如,您可以使用此方法获取随机用户:
$randomUser = DB::table('users')
->inRandomOrder()
->first();
来自文档:https ://laravel.com/docs/5.2/queries#ordering-grouping-limit-and-offset
您还可以使用 order_by 方法流利而雄辩,例如:
Posts::where_status(1)->order_by(DB::raw(''),DB::raw('RAND()'));
这是一个有点奇怪的用法,但有效。
编辑:正如@Alex所说,这种用法更干净,也有效:
Posts::where_status(1)->order_by(DB::raw('RAND()'));
您可以轻松地使用此命令:
// 问题 : 模型的名称
// 从 DB 中取 10 行 随机记录...
$questions = Question::orderByRaw('RAND()')->take(10)->get();
使用 Laravel 函数
ModelName::inRandomOrder()->first();
我更喜欢先指定或失败:
$collection = YourModelName::inRandomOrder()
->firstOrFail();
Laravel 有一个内置的方法来打乱结果的顺序。
这是文档中的引用:
shuffle()
shuffle 方法随机打乱集合中的项目:
$collection = collect([1, 2, 3, 4, 5]);
$shuffled = $collection->shuffle();
$shuffled->all();
// [3, 2, 5, 1, 4] - (generated randomly)
您可以在此处查看文档。
还有whereRaw('RAND()')
一个做同样的事情,然后你可以链接->get()
甚至->first()
疯狂添加->paginate(int)
。
在您的模型中添加以下内容:
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;
}
然后在路线/控制器
$data = YourModel::randomize(8)->get();
我有成千上万条记录的表,所以我需要一些快速的东西。这是我的伪随机行代码:
// count all rows with flag active = 1
$count = MyModel::where('active', '=', '1')->count();
// get random id
$random_id = rand(1, $count - 1);
// get first record after random id
$data = MyModel::where('active', '=', '1')->where('id', '>', $random_id)->take(1)->first();
试试这个代码!有用:
User::orderBy(DB::raw('RAND()'))->get();
以下是我在我的一个项目中获得随机结果的方法:
$products = Product::inRandomOrder()->limit(10);
10 - 要提取的随机记录数。
在 Laravel 7.x 及更高版本中,您可以这样做:
$data = Images::all()->random(4);