ChatGPT解决这个技术问题 Extra ChatGPT

Laravel - Eloquent 或 Fluent 随机行

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

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

有任何想法吗?

如果不执行至少两个查询,就没有最好的方法来做到这一点。

D
D Malan

Laravel >= 5.2:

User::inRandomOrder()->get();

或获取特定数量的记录

// 5 indicates the number of records
User::inRandomOrder()->limit(5)->get();
// get one random record
User::inRandomOrder()->first();

或对集合使用 random 方法:

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 随机行上的 this article。 Laravel 5.2 支持这一点,对于旧版本,没有比使用 RAW Queries 更好的解决方案。

编辑 1: 正如 Double Gras 所提到的,由于 this 更改,orderBy() 不允许除 ASC 或 DESC 之外的任何其他内容。我相应地更新了我的答案。

编辑 2: Laravel 5.2 最终为此实现了 a wrapper function。它被称为 inRandomOrder()


如果您想要单行,请将“get”替换为“first”。
对于 PostgreSQL 使用 'RANDOM()'
警告:在大型数据集上这非常慢,对我来说增加了大约 900 毫秒
我们可以分页吗?
您可以,但是在每个新页面上的排序都是随机的。这是没有意义的,因为它本质上与您按 F5 相同。
Y
Yevgeniy Afanasyev

这工作得很好,

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

您还可以更改随机函数中的参数以获取多条记录。

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


性能方面的缺点是检索所有记录。
这里 random 是在集合对象而不是 sql 查询上调用的。随机函数在php端运行
@astroanu 是的,但是要填充该集合,将查询所有行。
我可能是错的,但是当传递给随机函数的参数与集合的大小相同时,这似乎不起作用。
这不好......这样您就可以检索所有记录并获得随机记录。如果您的表有太多记录,这可能对您的应用程序不利。
G
Gras Double

tl;dr:它现在已在 Laravel 中实现,请参见下面的“编辑 3”。

遗憾的是,截至今天,->orderBy(DB::raw('RAND()')) 提议的解决方案存在一些注意事项:

它与数据库无关。例如 SQLite 和 PostgreSQL 使用 RANDOM()

更糟糕的是,这个解决方案不再适用,因为这个变化: $direction = strtolower($direction) == 'asc' ? “升序”:“降序”;


编辑: 现在您可以使用 orderByRaw() 方法:->orderByRaw('RAND()')。然而,这仍然不是 DB 无关的。

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

更新:这里是 GitHub 上关于此的 issue,以及我待处理的 pull request

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

您应该将 5.1 宏名称更改为 inRandomOrder 以便它向前兼容 ;) 详细信息,详细信息 :)
这正是我在将 5.1 项目迁移到 5.2 之前准备的一件事。
这是一个很好的答案。如果我能喜欢一个答案,我会的!
s
simhumileco

您可以使用:

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

T
Teodor Talov

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

所以,它应该是:

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

用户::orderBy(DB::raw('RAND()'))->get();
它的工作原理谢谢,但你能提供一些信息这是如何工作的吗?
你能再具体一点吗?什么样的信息?
T
Talles Airan

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

B
Bilal Gultekin

您还可以使用 order_by 方法流利而雄辩,例如:

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

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

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

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

这也很有效,而且更干净一些.. ->order_by(\DB::raw('RAND()'))
M
Manuel Azar

对于 Laravel 5.2 >=

使用雄辩的方法:

inRandomOrder()

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

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

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


Course::inRandomOrder()->take(20)->get();不适合我 - Find.php 第 219 行中的错误排序规范
这对于模型工厂或数据库播种很有用
E
Eric Aya

使用 Laravel 函数

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

s
simhumileco

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

// 问题 : 模型的名称 // 从 DB 中取 10 行 随机记录...

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

g
giovannipds

我更喜欢先指定或失败:

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

B
Brad Ahrens

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

这是文档中的引用:

shuffle()

shuffle 方法随机打乱集合中的项目:

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

$shuffled = $collection->shuffle();

$shuffled->all();

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

您可以看到 documentation here


N
Neto Braghetto

在您的模型中添加以下内容:

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

s
simhumileco

还有 whereRaw('RAND()') 可以做同样的事情,然后您可以链接 ->get()->first() 甚至疯狂添加 ->paginate(int)


D
Doc

我有成千上万条记录的表,所以我需要一些快速的东西。这是我的伪随机行代码:

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

这样做的问题是,如果有多个 id 大于 $count 的行,则只会检索其中的第一行,因此它也比任何其他行更有可能被检索。
佚名

试试这个代码!有用:

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

您的答案可以通过额外的支持信息得到改进。请edit添加更多详细信息,例如引用或文档,以便其他人可以确认您的答案是正确的。您可以找到有关如何写出好答案的更多信息in the help center
h
hackernewbie

以下是我在我的一个项目中获得随机结果的方法:

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

10 - 要提取的随机记录数。


C
Cody Gray

在 Laravel 7.x 及更高版本中,您可以这样做:

$data = Images::all()->random(4);

这应该非常小心地使用,因为整个数据将在随机过滤之前从数据库中获取。在大桌子上避免这种情况