如何在Laravel框架中使用Eloquent或Fluent选择随机行?
我知道,通过使用SQL,您可以按RAND()进行排序。但是,我希望获得随机行,而不是在初始查询之前对记录的数量进行计数。
什么好主意吗?
如何在Laravel框架中使用Eloquent或Fluent选择随机行?
我知道,通过使用SQL,您可以按RAND()进行排序。但是,我希望获得随机行,而不是在初始查询之前对记录的数量进行计数。
什么好主意吗?
当前回答
你可以使用:
ModelName::inRandomOrder()->first();
其他回答
还有whereRaw('RAND()')做同样的事情,然后你可以chain ->get()或->first(),甚至疯狂地添加->paginate(int)。
以下是我如何在我的一个项目中获得雄辩的随机结果:
$products = Product::inRandomOrder()->limit(10);
10 -随机记录的数量。
我有成千上万的记录表,所以我需要一些快速。这是我的伪随机行代码:
// 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();
在Laravel 4和5中,order_by被orderBy取代
所以,它应该是:
User::orderBy(DB::raw('RAND()'))->get();
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();