MySQL: Alternatives to ORDER BY RAND()

前端 未结 7 1857
情歌与酒
情歌与酒 2020-11-22 02:46

I\'ve read about a few alternatives to MySQL\'s ORDER BY RAND() function, but most of the alternatives apply only to where on a single random result is needed.

7条回答
  •  日久生厌
    2020-11-22 03:28

    UPDATE 2016

    This solution works best using an indexed column.

    Here is a simple example of and optimized query bench marked with 100,000 rows.

    OPTIMIZED: 300ms

    SELECT 
        g.*
    FROM
        table g
            JOIN
        (SELECT 
            id
        FROM
            table
        WHERE
            RAND() < (SELECT 
                    ((4 / COUNT(*)) * 10)
                FROM
                    table)
        ORDER BY RAND()
        LIMIT 4) AS z ON z.id= g.id
    

    note about limit ammount: limit 4 and 4/count(*). The 4s need to be the same number. Changing how many you return doesn't effect the speed that much. Benchmark at limit 4 and limit 1000 are the same. Limit 10,000 took it up to 600ms

    note about join: Randomizing just the id is faster than randomizing a whole row. Since it has to copy the entire row into memory then randomize it. The join can be any table that is linked to the subquery Its to prevent tablescans.

    note where clause: The where count limits down the ammount of results that are being randomized. It takes a percentage of the results and sorts them rather than the whole table.

    note sub query: The if doing joins and extra where clause conditions you need to put them both in the subquery and the subsubquery. To have an accurate count and pull back correct data.

    UNOPTIMIZED: 1200ms

    SELECT 
        g.*
    FROM
        table g
    ORDER BY RAND()
    LIMIT 4
    

    PROS

    4x faster than order by rand(). This solution can work with any table with a indexed column.

    CONS

    It is a bit complex with complex queries. Need to maintain 2 code bases in the subqueries

提交回复
热议问题