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

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

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

Java EE(Java Enterprise Edition)はJavaベースのテクノロジーとその相互運用の仕様をまとめたものです。サーバとクライアントのアーキテクチャを規定し、特定アプリケーションのクラス用に定義されたテクノロジー設定のプロファイルを使用します。

Q&A

解決済

3回答

17850閲覧

JPAでのPrimaryKeyが無くNULLのあるテーブルへのアクセスについて

hitokamu

総合スコア11

Java EE

Java EE(Java Enterprise Edition)はJavaベースのテクノロジーとその相互運用の仕様をまとめたものです。サーバとクライアントのアーキテクチャを規定し、特定アプリケーションのクラス用に定義されたテクノロジー設定のプロファイルを使用します。

0グッド

0クリップ

投稿2016/09/06 13:48

編集2016/09/08 02:01

###JPAでのPrimaryKeyが無くNULLのあるテーブルへのアクセスについて

S2DAOで構築されたシステムをJPAで置き換えを試みています。
テーブルの中にPrimaryKeyが無いものがあったのですが、稼働中のシステムの為データベースにPrimaryKeyを追加することはできません。
そこで全てのカラムをEmbeddableとして定義することとしたのですが、この方法ではデータの中に一つでもNULLが含まれる場合にDatabaseExceptionが発生してしました。
そもそもJPAではこのような使い方をすることが許されないのでしょうか?。

###発生している問題・エラーメッセージ
selectAllでAAA_TBLに対しSELECTを実行しようとすると、
カラムBBB,CCCにnullのデータが無い状態ではSELECTの実行結果を
取得できるのですが、nullが1件でもあると
「※ここで例外」とコメントを付けた箇所でDatabaseExceptionが発生してしまいます。

エラーメッセージ 2016-09-06 19:59:14,832 [http-nio-8080-exec-16] FATAL (AAA.java:48) - javax.persistence.PersistenceException: Exception [EclipseLink-6044] (Eclipse Persistence Services - 2.6.3.v20160428-59c81c5): org.eclipse.persistence.exceptions.QueryException Exception Description: The primary key read from the row [ArrayRecord( AAA_TBL.BBB => null AAA_TBL.CCC => null)] during the execution of the query was detected to be null. Primary keys must not contain null. Query: ReadAllQuery(name="AAATbl.findAll" referenceClass=AAATbl sql="SELECT BBB, CCC FROM AAA_TBL") ###該当のソースコード @Entity @Table(name = "AAA_TBL") @XmlRootElement @NamedQueries({ @NamedQuery(name = "AAA.findAll", query = "SELECT c FROM AAATbl c")}) public class AAATbl implements Serializable { private static final long serialVersionUID = 1L; @EmbeddedId protected AAATblPK aaaTblPK; public AAATbl() { } public AAATbl(AAATblPK aaaTblPK) { this.aaaTblPK= aaaTblPK; } public AAATbl(String bbb,String ccc) { this.aaaTblPK= new AAATblPK(bbb,ccc); } public AAATblPK getAAATblPK() { return this.aaaTblPK; } public void setAAATblPK(AAATblPK aaaTblPK) { this.aaaTblPK= aaaTblPK; } public class AAATblPK implements Serializable { @Size(max = 1 ) @Column(name = "BBB") private String bbb; @Size(max = 1 ) @Column(name = "CCC") private String ccc; public AAAPK() { } public AAAPK(String bbb,String ccc) { this.bbb = bbb; this.ccc = ccc; } public String getBbb() { return bbb; } public void setBbb(String bbb) { this.bbb = bbb; } public String getCcc() { return ccc; } public void setCcc(String Ccc) { this.ccc = ccc; } @Override public int hashCode() { int hash = 0; hash += (bbb != null ? bbb.hashCode() : 0); hash += (ccc != null ? ccc.hashCode() : 0); return hash; } @Override public boolean equals(Object object) { if (!(object instanceof AAATblPK)) { return false; } AAATblPK other = (AAATblPK) object; if ((this.bbb== null && other.bbb!= null) || (this.bbb!= null && !this.bbb.equals(other.bbb))) { return false; } if ((this.ccc== null && other.ccc!= null) || (this.ccc!= null && !this.ccc.equals(other.ccc))) { return false; } return true; } @Override public String toString() { return "tmp.AAATblPK[ bbb= "+ bbb + ",ccc=" + ccc +" ]"; } } public AAATbl[] selectAll() { try { Query query; query = em.createNamedQuery("AAATbl.findAll"); List<AAATbl> list; list = query.getResultList(); // ※ここで例外 AAATbl[] ret; ret = (AAATbl[]) list.toArray(new AAATbl[list.size()]); return ret; } catch (NoResultException e) { return null; } catch (PersistenceException | DatabaseException databaseEx) { logger.fatal(e); return null; } catch (Exception e) { logger.fatal(e); return null; } }

テーブル構造

実際のカラムは32個ありますが、"BBB","CCC"の2つに省略化して記載しています。

CREATE TABLE "DATABASE"."AAA_TBL" ( "BBB" VARCHAR2(1 BYTE), "CCC" VARCHAR2(1 BYTE) ) SEGMENT CREATION IMMEDIATE PCTFREE 10 PCTUSED 40 INITRANS 1 MAXTRANS 255 NOCOMPRESS NOLOGGING STORAGE(INITIAL 131072 NEXT 1048576 MINEXTENTS 1 MAXEXTENTS 2147483645 PCTINCREASE 0 FREELISTS 1 FREELIST GROUPS 1 BUFFER_POOL DEFAULT FLASH_CACHE DEFAULT CELL_FLASH_CACHE DEFAULT) TABLESPACE "USERS" ;

###試したこと
テーブルの値にNULL値が含まれない場合は値を取得できる事を確認しています。
@PrimaryKey(validation=IdValidation.NONE)を追加してもみたのですが、状況は変わりませんでした。

(2016/09/08追記)
最適解かどうかはわからないのですが、NamedQueryを使うのを諦め、NativeQueryでObject型とすればテーブルのデータを取り出せることは確認できました。

public AAATbl[] selectAll() { try { String sql = "SELECT BBB,CCC FROM AAA_TBL"; List<AAATbl> dest = new ArrayList<>(); List<Object[]> list = (List<Object[]>) em.createNativeQuery(sql).getResultList(); Iterator it = list.iterator(); while (it.hasNext()) { AAATbl data = new AAATbl(); Object[] obj = (Object[]) it.next(); int i = 0; data.bbb((String) obj[i++]); data.ccc((String) obj[i++]) dest.add(data); } AAATbl[] ret; ret = (AAATbl[]) dest.toArray(new AAATbl[dest.size()]); return ret; } catch (NoResultException e) { return null; } catch (PersistenceException | DatabaseException databaseEx) { logger.fatal(databaseEx); return null; } catch (Exception e) { logger.fatal(e); return null; } }

###補足情報(言語/FW/ツール等のバージョンなど)
環境はNetBeans(8.1)+TomEE(7.0.1)
JPAはEclipse Persistence Services - 2.6.3.v20160428-59c81c5を使用しています。
DataBaseは Oracle 11.2.0.1.0を使用しています。

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

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

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

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

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

退会済みユーザー

退会済みユーザー

2016/09/06 15:26

TomcatEE ? TomEE の間違い?それともEE自体が間違い?
hitokamu

2016/09/07 04:23

TomEEの誤記でしたので訂正しました。ご指摘ありがとうございます。
guest

回答3

0

自己解決

createNativeQueryの引数にクラスを指定せず、Object型として取り出した後にListに詰めることでデータを取り出すことは可能になりました。
ソースコードは「試したこと」の2016/09/08に追記したものとなります。

投稿2016/09/13 00:39

hitokamu

総合スコア11

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

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

0

JPAではEntityBeanはDBのTableデザインと一致していることが前提となります。EntityBeanのメンバ定義とTableのcolumn定義が不一致の場合、何らかの不具合が起きても不思議はないです。PrimalyKeyにnullが含まれるというのはDBとして矛盾なので、エラーとして弾いている可能性が考えられます。ご質問に対する解としてはできないのではないかと思います。

DB側が変更できない事情のようなので、EntityBean側のPrimalyKeyアノテーションを外すことになるかと思います。同一事象に遭遇していないので、試したことはないのですが。


EntityBean側にPrimalyKey指定が有るものと勘違いしていました。済みません。
PrimalyKey指定していないので、値がnullでも問題ないように思えます。Oracle接続の環境が作れないのでちょっと調べられません。解決につながるか分かりませんが、エラー内容やサーバーログなど提示してみてはどうでしょうか。関係ないかも知れませんが、varcher2のサイズがTableとEntityで異なるのが気になりました。


EmbeddedIdアノテーションは複合主キーの指定となるようです。

複合主キーの定義方法

このためAAATblPKがPrimalyKeyとされ、nullでExceptionとなったものと思われます。
@EmbeddedIdを使用するのをやめるのと同時にメンバ指定をサブクラスで指定するのをやめ、AAATblに直接bbb、cccのメンバを記載されてはどうでしょうか。

投稿2016/09/07 00:43

編集2016/09/08 02:44
BlueMoon

総合スコア1339

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

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

hitokamu

2016/09/08 02:16

大変申し訳ないです。varcher2のサイズは元のソースから質問の為編集した際の誤りです。 エラー内容も修正しています。 現時点でNativeQueryを使えば回避できそうなことまでは判明し、「試したこと」の欄に追記したのですが、最適かわからない状況です。
hitokamu

2016/09/08 11:09

@EmbeddedIdの使用を止めたとしても@Idか@IdClassを使用しない限り実行時にエラーとなってしまうので解決策にはなりえないようです。 Exception Description: Entity class [class AAATbl] has no primary key specified. It should define either an @Id, @EmbeddedId or an @IdClass. PrimaryKeyの無いテーブルにJPAでアクセスしなくてはならないこと自体が無理難題でしかなさそうなので、NativeQueryでの回避策を採ることにします。
BlueMoon

2016/09/08 11:54

了解です。回避策が有るようで何よりでした。
BlueMoon

2016/09/08 12:10

回避策をもって自己解決とされて良いかと思います。
hitokamu

2016/09/13 00:25

助言いただきありがとうございます。回避策を自己解決とします。
guest

0

column構成がわかれば解はでるきがするが
テーブル名はtable,システムを連想しそうなカラムにはcol01等にかえたcreate文をのせてみよう。

そのまままでもいいです。

投稿2016/09/06 23:08

退会済みユーザー

退会済みユーザー

総合スコア0

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

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

hitokamu

2016/09/07 04:48

助言していただきありがとうございます。カラム名は質問した際のまま(BBB,CCC)ですが、テーブル構造の欄を追記しました。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問