タイトル: Number of positional parameter types (1 does not match number of positional parameters (2)
SEOタイトル: Hibernate QueryException「Number of positional parameter types」の原因と対処(?パラメータ不一致)
| この記事の要点 |
|
エラー全文
net.sf.hibernate.QueryException: Number of positional parameter types (1) does not match number of positional parameters (2) [from User u where u.name = ? and u.deleted = ?]
カッコ内の数字 (1) が型を指定した数、(2) が HQL 中の ? の数。両者が食い違うと発生します。
発生原因
Hibernate の positional parameter(? による位置指定パラメータ)を使う場合、HQL/SQL に書いた ? の数と、Java 側で setXxx() を呼ぶ回数は完全一致が必要です。
| 状況 | HQL の ? 数 | setXxx() 呼出数 | 結果 |
|---|---|---|---|
| 正常 | 2 | 2 | OK |
| setXxx を 1 つ忘れた | 2 | 1 | QueryException |
HQL に ? 増やしたが set 忘れ | 3 | 2 | QueryException |
| set を余分に呼んだ | 1 | 2 | QueryException |
NG 例 / OK 例
NG: ? 2 個に対し setString 1 個
Query q = session.createQuery(
"from User u where u.name = ? and u.deleted = ?");
q.setString(0, "alice");
// ↑ deleted の ? が未バインド → QueryException
List users = q.list();
OK: ? 2 個に対し setXxx 2 個
Query q = session.createQuery(
"from User u where u.name = ? and u.deleted = ?");
q.setString(0, "alice");
q.setBoolean(1, false);
List users = q.list();
推奨: named parameter (:name) に書き換える
positional は数え間違いが起きやすい。名前付きパラメータを使うと安全で可読性も上がります。
// HQL: :name / :deleted を使う
Query q = session.createQuery(
"from User u where u.name = :name and u.deleted = :deleted");
q.setString("name", "alice");
q.setBoolean("deleted", false);
List users = q.list();
デバッグ手順
- 例外メッセージから HQL 全文を確認(スタックトレース末尾)
- HQL の
?の数を数える(エディタの検索カウント機能が便利) - Java 側の
setXxx()呼び出しを数える - 不一致箇所を特定 → 追加 or 削除 or named parameter に書き換え
hibernate.show_sql=trueで実際に発行された SQL を確認
関連エラー
- QueryParameterException: could not locate named parameter — 名前付きパラメータの名前が一致しない
- IllegalArgumentException: Parameter with that position does not exist — JPA でインデックス指定ミス
- SQLGrammarException — SQL 文法エラー(パラメータ数ではなく文法)