Laravel-EloquentまたはFluentのランダムな行


242

LaravelフレームワークでEloquentまたはFluentを使用してランダムな行を選択するにはどうすればよいですか?

SQLを使用すると、RAND()で順序を指定できることがわかります。ただし、最初のクエリの前にレコード数を数えずにランダムな行を取得したいと思います。

何か案は?


少なくとも2つのクエリを実行せずにこれを行う最善の方法はありません。
NARKOZ、

回答:


584

Laravel> = 5.2:

User::all()->random();
User::all()->random(10); // The amount of items you wish to receive

または

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

または特定の数のレコードを取得する

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

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はこれをサポートしています。古いバージョンの場合、RAWクエリを使用するよりも優れたソリューションはありません。

編集1: Double Grasで述べたように、orderBy()では、この変更以降、ASCまたはDESC以外は許可されません。私はそれに応じて私の答えを更新しました。

編集2: Laravel 5.2は最終的にこのためのラッパー関数を実装します。これはinRandomOrder()と呼ばれます。


81
単一の行が必要な場合は、「get」を「first」に置き換えます。
コリン価格

14
PostgreSQLの使用'RANDOM()'
dwenaus 2014年

2
警告:大規模なデータセット上で、これは私のために約900ミリ秒を追加し、非常に遅いです
S ...

3
これにページ番号を付けることはできますか?
Irfandi D. Vendy 2015

3
ただし、並べ替えは新しいページごとにランダムになります。F5キーを押すのと基本的に同じなので、これは意味がありません。
aebersold 2015

49

これはうまく機能し、

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

ランダム関数の引数を変更して、複数のレコードを取得することもできます。

注:巨大なデータがある場合は、最初にすべての行をフェッチしてからランダムな値を返すため、お勧めしません。


61
パフォーマンス面での欠点は、すべてのレコードが取得されることです。
Gras Double、

3
ここでは、SQLクエリではなくコレクションオブジェクトでランダムに呼び出されます。random関数はphp側で実行されます
astroanu

@astroanu正解ですが、そのコレクションを作成するために、すべての行がクエリされます。
MetalFrog 2015年

1
私は間違っている可能性がありますが、ランダム関数に渡されたパラメーターがコレクションのサイズと同じである場合、これは機能しないようです。
ブリンベイトマン2016年

これは良くありません...このようにして、すべてのレコードを取得し、ランダムなレコードを取得します。テーブルのレコードが多すぎると、アプリに悪影響を与える可能性があります。
アンダーソンシルバ

34

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]);
});

使用法:

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

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


編集3:最後に!Laravel 5.2.33(changelogPR# 13642)以降、ネイティブメソッドを使用できますinRandomOrder()

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

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

5.1マクロ名をinRandomOrderに変更して、上位互換性を持たせる必要があります;)詳細、詳細:)
Sander Visser

これは、5.1プロジェクトを5.2に移行する前に準備していたときに私がしたことの1つです。
Gras Double

これは素晴らしい答えです。答えが良かったら、そうしよう!
mwallisch

18

Laravel 4及び5order_byにより置換されていますorderBy

したがって、次のようになります。

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

ユーザー:: orderBy(DB :: raw( 'RAND()'))-> get();
Darius M.

1
うまくいきましたが、これがどのように機能するかについていくつか情報を提供できますか?
alayli 2014

あなたはもう少し具体的にできますか?どんな情報?
テオドールタロフ2014


9

Laravel 5.2の場合> =

Eloquentメソッドを使用します。

inRandomOrder()

inRandomOrderメソッドを使用すると、クエリ結果をランダムにソートできます。たとえば、このメソッドを使用してランダムなユーザーをフェッチできます。

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

ドキュメントから:https : //laravel.com/docs/5.2/queries#ordering-grouping-limit-and-offset


コース:: inRandomOrder()-> take(20)-> get(); 私のために働いていない- Find.phpライン219で悪いソート指定
MJ

1
これは、モデルファクトリまたはdbシードに役立ちます
Saleh Mahmood

8

次のような流暢で雄弁なorder_byメソッドを使用することもできます。

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

これは少し奇妙な使い方ですが、動作します。

編集:@アレックスが言ったように、この使用法はよりクリーンであり、うまくいきます:

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

3
これも同様に機能し、少しクリーンになります
。-


3

このコマンドは簡単に使用できます。

//質問:モデル名
// DBから10行を取得して、レコードをシャッフルします...

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

3

私は最初に指定するか、失敗することを好みます:

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

3

Laravelには、結果の順序を並べ替える組み込みメソッドがあります。

これはドキュメントからの引用です:

shuffle()

シャッフルメソッドは、コレクション内のアイテムをランダムにシャッフルします。

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

$shuffled = $collection->shuffle();

$shuffled->all();

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

こちらドキュメントをご覧ください


2

あなたのモデルでこれを追加してください:

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

2

ありwhereRaw('RAND()')ますがチェーン、その後、同じことをしていることができ->get()たり->first()、あるいは夢中になると追加が->paginate(int)


0

数千のレコードを含むテーブルがあるので、何か高速なものが必要です。これは、疑似ランダム行の私のコードです:

// 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、最初の行だけが取得されるため、他の行よりも取得される可能性が高いことです。
kemika
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.