汎用関数の問題(続き)
では続けて、次の問題も解いてみてください。
問3
次を確認してください。
結果を返す(実行エラーにならない)SQLをすべて選びなさい
- a. SELECT NVL(comm,'No Data') FROM emp;
- b. SELECT NVL(comm,0) FROM emp;
- c. SELECT NVL2(comm,comm,'No Data') FROM emp;
- d. SELECT NVL2(comm,TO_CHAR(comm),'No Data') FROM emp;
引数のデータ型を問う問題です。あまり気にしていないかもしれませんが、NVL
関数に渡す第1引数と第2引数は、データ型が一致していなければいけません。comm
列はNUMBER
型、選択肢aの第2二引数'No Data'
は文字型なので、実行時エラーになります。選択肢bは「0」という数値だから結果を返すので、正解です。
SQL> --選択肢a SQL> SELECT NVL(comm,'No Data') FROM emp; SELECT NVL(comm,'No Data') FROM emp * 行1でエラーが発生しました。: ORA-01722: 数値が無効です。 SQL> --選択肢b SQL> SELECT NVL(comm,0) FROM emp; NVL(COMM,0) ----------- 0 300 500 0 1400 0 0 0 0 0 0 0 12行が選択されました。
NVL2
関数の場合は、第2引数と第3引数が同じ型でなければいけません。結果として表示されるのは、第2引数と第3引数ですよね、「1つの列に一緒に表示する以上、データ型は同じにしておいてもらわないと困るよ」というエラーだと思えば納得するでしょ。
「比較する第1引数と第2引数の型は同じでなくていいのか?」そう言いたい気持ちはわかりますが、Oracleの気持ちになれば「同じにするのは、比較しようとしているあなたの責任。違うなら、一致しないわよって私は結果を返すだけだから」と考えていただければ、皆さんの記憶に残るのではないでしょうか。
SQL> --選択肢c SQL> SELECT NVL2(comm,comm,'No Data') FROM emp; SELECT NVL2(comm,comm,'No Data') FROM emp * 行1でエラーが発生しました。: ORA-01722: 数値が無効です。 SQL> --選択肢d SQL> SELECT NVL2(comm,TO_CHAR(comm),'No Data') FROM emp; NVL2(COMM,TO_CHAR(COMM),'NODATA') ---------------------------------------- No Data 300 500 No Data 1400 No Data No Data No Data 0 No Data No Data No Data 12行が選択されました。
NULL
がらみの関数にはCOALESCE
があります。「引数として指定した複数の列の中で、最初に検出されたNULL
でない値を戻す」関数です。COALESCE
に限らず、関数は使えるだろう場面を想像できれば、覚えられると思います。
例えば、COALESCE
なら、こんなのはどうですか?(すべての列のデータ型は同じとします)
SELECT COALESCE(email,会社電話,自宅電話,携帯電話,住所,'連絡先不明') AS 連絡方法 FROM 顧客情報
emailがわからない(値がNULL
)ならば、会社の電話番号……がわからないならば自宅……がわからないならば……全部わからなければ(email列から住所列まですべての列の値がNULL
)、「連絡先不明」と出力するというSQL文です。