質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.48%
MySQL

MySQL(マイエスキューエル)は、TCX DataKonsultAB社などが開発するRDBMS(リレーショナルデータベースの管理システム)です。世界で最も人気の高いシステムで、オープンソースで開発されています。MySQLデータベースサーバは、高速性と信頼性があり、Linux、UNIX、Windowsなどの複数のプラットフォームで動作することができます。

Q&A

解決済

4回答

3127閲覧

ランダムな条件のSELECTの結果

XCUBE

総合スコア101

MySQL

MySQL(マイエスキューエル)は、TCX DataKonsultAB社などが開発するRDBMS(リレーショナルデータベースの管理システム)です。世界で最も人気の高いシステムで、オープンソースで開発されています。MySQLデータベースサーバは、高速性と信頼性があり、Linux、UNIX、Windowsなどの複数のプラットフォームで動作することができます。

6グッド

1クリップ

投稿2015/12/18 08:47

編集2015/12/18 08:52

なぜそうなるのかわかなくなったので教えてください。
MariaDBなんですが、phpMyAdminを使ってSELECTした結果がなぜそうなるのかわからないんです。

pointテーブルに2件のデータが入っています。
このテーブルのプライマリキーidカラムはtinyint(4)で2件それぞれ0と1が入っています。

次のSQLを実行します。

SELECT * FROM point WHERE id = (FLOOR(RAND() * 100) % 2)

式(FLOOR(RAND() * 100) % 2)が0か1になるようにしたつもりなのですが
何回か行った結果は
id=0が取得される場合
id=1が取得される場合
2件とも取得できる場合
どちらも取得できない場合
となりました。
この4パターンが出る確率はほぼ同じです。

なぜ2件取得されるのか?、どちらも取得できないのか?
そうなる理由が解りません。どうかお教えください。

環境
OS archlinux 4.2.5-1
DB mysqld 10.1.9-MariaDB

izkn, mhashi, anaprestoo, arly_times, TetsuyaZama👍を押しています

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答4

0

ベストアンサー

MySQLのリファレンスマニュアル(この辺にMySQL/MariaDBで差異はないでしょう)によると:

WHERE 句内の RAND() は、WHERE が実行されるたびに再評価されます。

とありますね。XCUBEさんの現在の想定だと

SELECT * FROM point WHERE 0 = (FLOOR(RAND() * 100) % 2)

の出力結果は、「0行」か「2行両方」のいずれかになるのではないでしょうか。
(が、実際にはそうではない、すなわち再評価されている)

投稿2015/12/18 10:50

編集2015/12/18 10:51
退会済みユーザー

退会済みユーザー

総合スコア0

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

XCUBE

2015/12/21 00:42

そういうことだったんですね。 私が一番多用するSQL ServerではRAND()の値は全ての行に対して一定となるので このようなことは全く想定していませんでした。 他の方も同様に回答いただきありがとうございました。
退会済みユーザー

退会済みユーザー

2015/12/21 02:03

マニュアルに敢えてこのように書かれているからには他のRDBMSだと違うのだろうな、と考え、PostgreSQL9.3, Oracle11g, sqlite3でも試してみたのですが、MySQLと同様の挙動になったので、あれ…?と思っていたところでしたw。なるほど!SQL Serverがそれだったのですね。普段あまりWindowsを使わないのですっかり失念しておりました…
XCUBE

2015/12/21 04:14

そういえば数年前にMySQLである検定の模擬試験のWebサービスの開発をしたとき 問題集が登録されたテーブルから無作為に30問出題するのに 「ORDER BY RAND() LIMIT 30」 で簡単に実装できたことを今思い出しました。私もすっかり失念しておりましたw
guest

0

1行毎にFLOOR(RAND() * 100) % 2が実行されているからです。
次のようなSQL文を発行するとよくわかります。

sql

1SELECT id, RAND() FROM point;

投稿2015/12/18 10:17

anaprestoo

総合スコア199

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

KiyoshiMishima

2015/12/18 12:53

言われ通りですね(^ ^) 乱数の評価の問題です 抽選ゲームとかの仕組みが理解出来れば、わかると思いますよ
guest

0

このSQLですが、設計中の画面のデモを行うため、表示するデータのパターンをDBに登録し、一定間隔でランダムに変化させたいために考えました。
想定ではランダムに一行だけを取得したかったのですが、ご回答いただいた通り、これではだめなので、ランダムな仮想テーブルにすることで無事解決しましたのでご報告させていただきます。

SQL

1SELECT a.* FROM point a INNER JOIN (SELECT (FLOOR(RAND() * 100) % 2) id) t ON a.id = t.id

投稿2015/12/21 01:38

編集2015/12/21 01:40
XCUBE

総合スコア101

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

0

これは RAND() がレコードごとに評価されるからだからかと思いますよ。

投稿2015/12/18 10:17

WataruTerada

総合スコア70

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.48%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問