Laravel - Eloquent或Fluent随机排

Dig*_*lWM 216 php fluent laravel eloquent

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

我知道通过使用SQL,您可以通过RAND()进行排序.但是,我想获取随机行而不计算初始查询之前的记录数.

有任何想法吗?

aeb*_*old 519

Laravel> = 5.2:

User::all()->random();
User::all()->random(10); // The amount of items you wish to receive
Run Code Online (Sandbox Code Playgroud)

Laravel 4.2.7 - 5.1:

User::inRandomOrder()->get();
Run Code Online (Sandbox Code Playgroud)

Laravel 4.0 - 4.2.6:

//5 indicates the number of records
User::inRandomOrder()->limit(5)->get();
Run Code Online (Sandbox Code Playgroud)

Laravel 3:

User::orderByRaw("RAND()")->get();
Run Code Online (Sandbox Code Playgroud)

查看有关MySQL随机行的这篇文章.Laravel 5.2支持这一点,对于旧版本,使用RAW查询没有更好的解决方案.

编辑1:如双节,排序依据()提到不允许因为别的然后ASC或DESC 这个变化.我相应地更新了答案.

编辑2: Laravel 5.2最终为此实现了一个包装函数.它被称为inRandomOrder().

  • 如果您想要一行,请将'get'替换为'first'. (77认同)
  • 对于PostgreSQL使用''RANDOM()'` (14认同)
  • 我们可以分页吗? (3认同)
  • 但是,您可以在每个新页面上随机排序.这没有任何意义,因为它与按F5基本相同. (3认同)
  • 警告:在大型数据集上,这非常慢,对我而言大约需要900毫秒 (2认同)

man*_*ish 47

这很好用,

$model=Model::all()->random(1)->first();
Run Code Online (Sandbox Code Playgroud)

您还可以在随机函数中更改参数以获取多个记录.

注意:如果您有大量数据,则不推荐使用,因为这将首先获取所有行,然后返回随机值.

  • 性能方面的缺点是检索所有记录. (57认同)
  • 在这里,对集合对象而不是sql查询调用random。随机函数在php端运行 (2认同)

Gra*_*ble 32

tl; dr:现在已经在Laravel中实现了,请参阅下面的"编辑3".


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

  • 它不是DB不可知的.例如SQLite和PostgreSQL使用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]);
});
Run Code Online (Sandbox Code Playgroud)

用法:

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

DB::table('users')->where('active', 1)->orderByRandom()->limit(10)->get();
Run Code Online (Sandbox Code Playgroud)


编辑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();
Run Code Online (Sandbox Code Playgroud)


小智 24

很简单,只需检查你的 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();
Run Code Online (Sandbox Code Playgroud)

或使用随机方法进行集合:

User::all()->random();
User::all()->random(10); // The amount of items you wish to receive
Run Code Online (Sandbox Code Playgroud)

Laravel 4.2.7 - 5.1:

 User::orderByRaw("RAND()")->get();
Run Code Online (Sandbox Code Playgroud)

Laravel 4.0 - 4.2.6:

 User::orderBy(DB::raw('RAND()'))->get();
Run Code Online (Sandbox Code Playgroud)

拉拉维尔 3:

 User::order_by(DB::raw('RAND()'))->get();
Run Code Online (Sandbox Code Playgroud)


Teo*_*lov 18

Laravel 4和5中,order_by它被替换为orderBy

所以,它应该是:

User::orderBy(DB::raw('RAND()'))->get();
Run Code Online (Sandbox Code Playgroud)


sim*_*eco 14

你可以使用:

ModelName::inRandomOrder()->first();
Run Code Online (Sandbox Code Playgroud)


Bil*_*kin 8

您还可以使用具有流利和雄辩的order_by方法,如:

Posts::where_status(1)->order_by(DB::raw(''),DB::raw('RAND()')); 
Run Code Online (Sandbox Code Playgroud)

这有点奇怪的用法,但有效.

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

Posts::where_status(1)->order_by(DB::raw('RAND()'));
Run Code Online (Sandbox Code Playgroud)

  • 这也有效,而且更干净.. - > order_by(\ DB :: raw('RAND()')) (3认同)

Man*_*zar 8

对于Laravel 5.2> =

使用Eloquent方法:

inRandomOrder()
Run Code Online (Sandbox Code Playgroud)

inRandomOrder方法可用于随机对查询结果进行排序.例如,您可以使用此方法来获取随机用户:

$randomUser = DB::table('users')
            ->inRandomOrder()
            ->first();
Run Code Online (Sandbox Code Playgroud)

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

  • 这对于模型工厂或数据库播种很有用 (2认同)