Laravel -雄辩或流利随机行

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

我知道,通过使用SQL,您可以按RAND()进行排序。然而,我想让随机行没有在初始查询之前对记录的数量进行计数。

什么好主意吗?

369163 次浏览

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();

Laravel 3:

User::order_by(DB::raw('RAND()'))->get();

在MySQL随机行上检查这篇文章。Laravel 5.2支持这一点,对于旧版本,没有比使用原始查询更好的解决方案了。

编辑1:正如Double Gras所提到的,自从发生变化以来,orderBy()不允许ASC或DESC以外的任何内容。我相应地更新了我的答案。

Laravel 5.2最终为此实现了包装器函数。它叫做inRandomOrder ()

你也可以使用order_by方法,如:

Posts::where_status(1)->order_by(DB::raw(''),DB::raw('RAND()'));

这是一个有点奇怪的用法,但是有用。

编辑:正如@Alex所说,这种用法更简洁,也有效:

Posts::where_status(1)->order_by(DB::raw('RAND()'));

Laravel 4和5中,order_byorderBy取代

所以,它应该是:

User::orderBy(DB::raw('RAND()'))->get();

这很好,

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

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

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

tl;博士:它现在被实现到Laravel中,参见下面的“编辑3”。


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

  • 它不是数据库不可知论者。例如,SQLite和PostgreSQL使用RANDOM()
  • 更糟糕的是,这个解决方案不再适用,因为这种变化:

    $direction = strtolower($direction) == 'asc' ? 'asc' : 'desc'; < / p >

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

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

更新:这里是关于这个在GitHub上的问题,和我悬而未决的把请求

< p > < br > 编辑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();
< p > < br > edit 3:终于!由于Laravel 5.2.33 (更新日志公关# 13642),您可以使用本机方法inRandomOrder():

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


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

还有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;
}

然后是route/controller

$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();

对于Laravel 5.2 >=

使用Eloquent方法:

inRandomOrder()

可以使用inRandomOrder方法对查询结果进行随机排序。例如,你可以使用这个方法获取一个随机用户:

$randomUser = DB::table('users')
->inRandomOrder()
->first();

from docs: https://laravel.com/docs/5.2/queries#ordering-grouping-limit-and-offset

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

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

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

你可以使用:

ModelName::inRandomOrder()->first();

使用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)

你可以看到文件在这里

以下是我如何在我的一个项目中获得雄辩的随机结果:

$products           =  Product::inRandomOrder()->limit(10);

10 -随机记录的数量。

这很简单,只要检查你的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();

Laravel 3:

 User::order_by(DB::raw('RAND()'))->get();

试试这段代码!工作原理:

  User::orderBy(DB::raw('RAND()'))->get();