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

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

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

Javaは、1995年にサン・マイクロシステムズが開発したプログラミング言語です。表記法はC言語に似ていますが、既存のプログラミング言語の短所を踏まえていちから設計されており、最初からオブジェクト指向性を備えてデザインされています。セキュリティ面が強力であることや、ネットワーク環境での利用に向いていることが特徴です。Javaで作られたソフトウェアは基本的にいかなるプラットフォームでも作動します。

MyBatis

MyBatisはJavaや.NET Frameworkでなどで使用できる、SQL文や、ストアドプロシージャをオブジェクトと紐付けるO/Rマッピングフレームワークです。

Spring Boot

Spring Bootは、Javaのフレームワークの一つ。Springプロジェクトが提供する様々なフレームワークを統合した、アプリケーションを高速で開発するために設計されたフレームワークです。

Q&A

解決済

2回答

9005閲覧

[MyBatis] ListのNullをifでスキップしたい

sui_15

総合スコア24

Java

Javaは、1995年にサン・マイクロシステムズが開発したプログラミング言語です。表記法はC言語に似ていますが、既存のプログラミング言語の短所を踏まえていちから設計されており、最初からオブジェクト指向性を備えてデザインされています。セキュリティ面が強力であることや、ネットワーク環境での利用に向いていることが特徴です。Javaで作られたソフトウェアは基本的にいかなるプラットフォームでも作動します。

MyBatis

MyBatisはJavaや.NET Frameworkでなどで使用できる、SQL文や、ストアドプロシージャをオブジェクトと紐付けるO/Rマッピングフレームワークです。

Spring Boot

Spring Bootは、Javaのフレームワークの一つ。Springプロジェクトが提供する様々なフレームワークを統合した、アプリケーションを高速で開発するために設計されたフレームワークです。

0グッド

0クリップ

投稿2022/02/24 22:40

編集2022/02/24 23:26

閲覧ありがとうございます。
MyBatisでのnullの挙動で困っているので、ご教授お願い致します。
具体的には、list(1つ目のnull)をMapperでnull判定すると
org.springframework.jdbc.BadSqlGrammarException at SecondTest.java:38
Caused by: org.postgresql.util.PSQLException at SecondTest.java:38
というエラーが出てしまいます。
List.ofでfoodsMapper.findByQuery(new FoodsQuery(List.of("野菜"), null, null))のようにnullでない値を入力すると正常に動作します。
また、ほかの部分をnullにして、ifで書いた場合も正常に動作します。
つまり、Listの部分のみnullのスキップができない状態です。

説明に不備等ございましたら、コメントお願いいたします。
よろしくお願いします。ありがとう

java

1public class SecondTest{ 2 @Autowired 3 private FoodsMapper foodsMapper; 4 5 @Test 6 void test(){ 7 System.out.println(foodsMapper.findByQuery(new FoodsQuery(null, null, "アイス"))); //エラー 8System.out.println(foodsMapper.findByQuery(new FoodsQuery(List.of("氷菓"), null, "アイス"))); //成功する 9 } 10}

mapper.xml

1<?xml version="1.0" encoding="UTF-8"?> 2<!DOCTYPE mapper PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN" 3 "http://mybatis.org/dtd/mybatis-3-mapper.dtd"> 4<mapper namespace="curriculum.mapper.FoodsMapper"> 5 <select id="findByQuery" resultType="curriculum.domain.FoodsQuery"> 6 Select 7 * 8 From 9 foods 10 Where 11 <if test = "types != null" > 12 AND type IN 13 <foreach item="item" index="index" collection="types" 14 open="(" separator="," close=")" > 15 #{item} 16 </foreach> 17 </if> 18 <if test="limit != null"> 19 AND price <![CDATA[ <= ]]> #{limit} 20 </if> 21 <if test="word != null"> 22 AND name LIKE CONCAT('%', #{word}, '%') 23 </if> 24 </select> 25</mapper>

java

1@Mapper 2public interface FoodsMapper{ 3 List<Map<String, Object>> findByQuery(FoodsQuery param); 4}

java

1@Data 2public class FoodsQuery { 3 @Column(nullable = false) 4 public List<String> types; 5 public Integer limit; 6 public String word; 7 8 public FoodsQuery(){ 9 } 10 11 public FoodsQuery(List<String> types, Integer limit, String word) { 12 this.types = types; 13 this.limit = limit; 14 this.word = word; 15 } 16}

psql

1create table foods 2( 3 id SERIAL, 4 name text NOT NULL, 5 type text NOT NULL, 6 price integer NOT NULL 7); 8 9insert into foods (name, type, price) 10values ('りんご', '果物', 100), 11 ('いちご', '果物', 500), 12 ('西瓜', '果物', 2000), 13 ('じゃがいも', '野菜', 150), 14 ('白菜', '野菜', 300), 15 ('かぼちゃ', '野菜', 1000), 16 ('チョコレート', '菓子', 200), 17 ('バニラアイス', '氷菓', 300), 18 ('抹茶アイス', '氷菓', 300), 19 ('いちごアイス', '氷菓', 300), 20 ('雪見だいふく', '氷菓', 200);

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

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

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

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

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

m.ts10806

2022/02/24 22:42

テーブル定義の問題では。 NULLを受け入れるか、受け入れないか なので。
sui_15

2022/02/24 22:59

m.ts10806様 ご回答ありがとうございます。 @Column(nullable = false)をテーブルに記載しましたが、うまくいきませんでした。
m.ts10806

2022/02/24 23:17

質問本文編集して一通りコード提示してください。 あと「うまくいかない」だと何が起きてるか分からないので、具体的に記載を。
sui_15

2022/02/24 23:22

m.ts10806様 一通りコードを提示させていただきました。 よろしくお願いします。
guest

回答2

0

ベストアンサー

System.out.println(foodsMapper.findByQuery(new FoodsQuery(null, null, "アイス"))); //エラー 三つ目のword以外がnullだと、こういうSQLになるのでは?

sql

1 Select 2 * 3 From 4 foods 5 Where 6 AND name LIKE CONCAT('%', #{word}, '%')

... where and ...はおかしいのではないかと。

けれど、System.out.println(foodsMapper.findByQuery(new FoodsQuery(List.of("氷菓"), null, "アイス"))); //成功する だとしても、

sql

1 Select 2 * 3 From 4 foods 5 Where 6 AND type IN

だから、うまくはいかなないような...。

動的 SQLをみると、

xml

1<select id="findActiveBlogWithTitleLike" 2 resultType="Blog"> 3 SELECT * FROM BLOG 4 WHERE state = ‘ACTIVE’ 5 <if test="title != null"> 6 AND title like #{title} 7 </if> 8</select>

としているので、state = ‘ACTIVE’代わりにこうすればいいのかも。

xml

1 <select id="findByQuery" resultType="curriculum.domain.FoodsQuery"> 2 Select 3 * 4 From 5 foods 6 Where 71 = 1 8 <if test = "types != null" > 9 AND type IN

投稿2022/02/25 00:03

shiketa

総合スコア4061

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

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

sui_15

2022/02/25 00:19

shiketa様 ご回答ありがとうございます。 whereのあとに1=1を記載したら、求める回答を得ることができました。 本当にありがとうございます。 この、1=1というのはどういうことなのでしょうか?
KoichiSugiyama

2022/02/25 03:35

shiketaさんの解説にも書いてありますよ。 >三つ目のword以外がnullだと、こういうSQLになるのでは? 条件によってSQLとして成り立たないものができてしまうので、1 = 1という常に成立する条件とANDを取る形に調整してあるということです。
shiketa

2022/02/25 04:39

です。 もし、全部の条件がnullだったらこうなる。`select * from foords where;` 全部の条件がnullだったら`where`句も除かないといけない。めんどうくさい。そのための`1=1`。`true`でもいいかもしれない。
sui_15

2022/03/02 21:13

Sugiyama様、shiketa様 返信遅くなってしまいすみません。 とても参考になりました。 また機会がありましたら、その時は宜しくお願い致します。
guest

0

<where>を利用されるとよいかと思います。

xml

1<where> 2 <if test = "types != null"> 3 type IN <foreach item="item" index="index" collection="types" open="(" separator="," close=")">#{item}</foreach> 4 </if> 5 <if test="limit != null"> 6 AND price <![CDATA[ <= ]]> #{limit} 7 </if> 8 <if test="word != null"> 9 AND name LIKE CONCAT('%', #{word}, '%') 10 </if> 11</where>

参考: https://mybatis.org/mybatis-3/ja/dynamic-sql.html

投稿2022/02/25 05:20

neko_the_shadow

総合スコア2349

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

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

sui_15

2022/03/02 21:15

neko様 ご回答ありがとうございます。 そのようなwhereの書き方もあるのですね、参考になります。 次、またこのようなコードを書く際は試してみます!
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.35%

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

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

質問する

関連した質問