2016-12-28 3 views
0

私は、GAEでホストされているバックエンドを持つAndroidアプリケーションを構築しています。今は、Objectifyを使用してPOJOエンティティをデータストアに永続化しようとしています。以下は私が持続しようとしているPOJOです。Objectifyの多態性:SaveExceptionの "idは0にできません"

@Entity 
public class GenericLearnerProfile { 

    private String mName; 
    @Id 
    private String mEmailID; 
    private String mPhoneNo; 
    private String mImagePath; 
    private String mCurrentStatus; 
    private String mPassword; 

    //Default constructor required for persistence(serialization) 
    public GenericLearnerProfile(){} 
    //.......Other Getters, Setters and constructors follow 
} 

@Entity 
public class TutorProfile extends GenericLearnerProfile { 

    private EducationalQualification[] educationalQualifications; 
    private Occupation occupation; 

    //Default constructor required for persistence(serialization) 
    public TutorProfile(){} 
    //.......Other Getters, Setters and constructors follow 
} 

ご覧のとおり、非常に単純な階層です。

問題:私はちょうどGenericLearnerProfileを保持しようとすると、それが成功しTutorProfileではなく起こる、それは次の例外をスローします:

com.googlecode.objectify.SaveException: Error saving [email protected]: id cannot be zero 
    at com.googlecode.objectify.impl.EntityMetadata.save(EntityMetadata.java:95) 
    at com.googlecode.objectify.impl.WriteEngine.save(WriteEngine.java:73) 
    at com.googlecode.objectify.impl.SaverImpl.entities(SaverImpl.java:60) 
    at com.googlecode.objectify.impl.SaverImpl.entity(SaverImpl.java:35) 
    at com.learncity.backend.persistence.TutorProfileEndpoint.insert(TutorProfileEndpoint.java:83) 
    at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method) 
    at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62) 
    at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43) 
    at java.lang.reflect.Method.invoke(Method.java:498) 
    at com.google.appengine.tools.development.agent.runtime.Runtime.invoke(Runtime.java:130) 
    at com.google.api.server.spi.SystemService.invokeServiceMethod(SystemService.java:363) 
    at com.google.api.server.spi.SystemServiceServlet.execute(SystemServiceServlet.java:113) 
    at com.google.api.server.spi.SystemServiceServlet.doPost(SystemServiceServlet.java:71) 
    at javax.servlet.http.HttpServlet.service(HttpServlet.java:637) 
    at javax.servlet.http.HttpServlet.service(HttpServlet.java:717) 
    at org.mortbay.jetty.servlet.ServletHolder.handle(ServletHolder.java:511) 
    at org.mortbay.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1166) 
    at com.googlecode.objectify.ObjectifyFilter.doFilter(ObjectifyFilter.java:48) 
    at org.mortbay.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1157) 
    at com.google.appengine.api.socket.dev.DevSocketFilter.doFilter(DevSocketFilter.java:74) 
    at org.mortbay.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1157) 
    at com.google.appengine.tools.development.ResponseRewriterFilter.doFilter(ResponseRewriterFilter.java:128) 
    at org.mortbay.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1157) 
    at com.google.appengine.tools.development.HeaderVerificationFilter.doFilter(HeaderVerificationFilter.java:34) 
    at org.mortbay.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1157) 
    at com.google.appengine.api.blobstore.dev.ServeBlobFilter.doFilter(ServeBlobFilter.java:63) 
    at org.mortbay.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1157) 
    at com.google.apphosting.utils.servlet.TransactionCleanupFilter.doFilter(TransactionCleanupFilter.java:50) 
    at org.mortbay.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1157) 
    at com.google.appengine.tools.development.StaticFileFilter.doFilter(StaticFileFilter.java:125) 
    at org.mortbay.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1157) 
    at com.google.appengine.tools.development.DevAppServerModulesFilter.doDirectRequest(DevAppServerModulesFilter.java:366) 
    at com.google.appengine.tools.development.DevAppServerModulesFilter.doDirectModuleRequest(DevAppServerModulesFilter.java:349) 
    at com.google.appengine.tools.development.DevAppServerModulesFilter.doFilter(DevAppServerModulesFilter.java:116) 
    at org.mortbay.jetty.servlet.ServletHandler$CachedChain.doFilter(ServletHandler.java:1157) 
    at org.mortbay.jetty.servlet.ServletHandler.handle(ServletHandler.java:388) 
    at org.mortbay.jetty.security.SecurityHandler.handle(SecurityHandler.java:216) 
    at org.mortbay.jetty.servlet.SessionHandler.handle(SessionHandler.java:182) 
    at org.mortbay.jetty.handler.ContextHandler.handle(ContextHandler.java:765) 
    at org.mortbay.jetty.webapp.WebAppContext.handle(WebAppContext.java:418) 
    at com.google.appengine.tools.development.DevAppEngineWebAppContext.handle(DevAppEngineWebAppContext.java:98) 
    at org.mortbay.jetty.handler.HandlerWrapper.handle(HandlerWrapper.java:152) 
    at com.google.appengine.tools.development.JettyContainerService$ApiProxyHandler.handle(JettyContainerService.java:513) 
    at org.mortbay.jetty.handler.HandlerWrapper.handle(HandlerWrapper.java:152) 
    at org.mortbay.jetty.Server.handle(Server.java:326) 
    at org.mortbay.jetty.HttpConnection.handleRequest(HttpConnection.java:542) 
    at org.mortbay.jetty.HttpConnection$RequestHandler.content(HttpConnection.java:938) 
    at org.mortbay.jetty.HttpParser.parseNext(HttpParser.java:755) 
    at org.mortbay.jetty.HttpParser.parseAvailable(HttpParser.java:218) 
    at org.mortbay.jetty.HttpConnection.handle(HttpConnection.java:404) 
    at org.mortbay.io.nio.SelectChannelEndPoint.run(SelectChannelEndPoint.java:409) 
    at org.mortbay.thread.QueuedThreadPool$PoolThread.run(QueuedThreadPool.java:582) 
Caused by: java.lang.IllegalArgumentException: id cannot be zero 
    at com.google.appengine.api.datastore.KeyFactory.createKey(KeyFactory.java:52) 
    at com.google.appengine.api.datastore.KeyFactory.createKey(KeyFactory.java:47) 
    at com.googlecode.objectify.util.DatastoreUtils.createKey(DatastoreUtils.java:77) 
    at com.googlecode.objectify.impl.KeyMetadata.getRawKey(KeyMetadata.java:187) 
    at com.googlecode.objectify.impl.translate.EntityCreator.save(EntityCreator.java:54) 
    at com.googlecode.objectify.impl.translate.EntityCreator.save(EntityCreator.java:16) 
    at com.googlecode.objectify.impl.translate.ClassTranslator.saveSafe(ClassTranslator.java:136) 
    at com.googlecode.objectify.impl.translate.ClassTranslator.saveSafe(ClassTranslator.java:21) 
    at com.googlecode.objectify.impl.translate.NullSafeTranslator.save(NullSafeTranslator.java:25) 
    at com.googlecode.objectify.impl.translate.ArrayTranslatorFactory$1.save(ArrayTranslatorFactory.java:83) 
    at com.googlecode.objectify.impl.translate.ArrayTranslatorFactory$1.save(ArrayTranslatorFactory.java:39) 
    at com.googlecode.objectify.impl.PropertyPopulator.save(PropertyPopulator.java:135) 
    at com.googlecode.objectify.impl.translate.ClassPopulator.save(ClassPopulator.java:153) 
    at com.googlecode.objectify.impl.translate.ClassTranslator.saveSafe(ClassTranslator.java:138) 
    at com.googlecode.objectify.impl.translate.ClassTranslator.saveSafe(ClassTranslator.java:21) 
    at com.googlecode.objectify.impl.translate.NullSafeTranslator.save(NullSafeTranslator.java:25) 
    at com.googlecode.objectify.impl.EntityMetadata.save(EntityMetadata.java:89) 
    ... 51 more 

注:

  • すべての埋め込みオブジェクトはIDとして長いです。
  • 私はそれらを長くすると助けになりません。次に、エラーを表示します。 "idはnullにはできません"
  • 組み込みオブジェクトのlong IDは手動で設定する必要はありません。永続性)

誰も私にこの例外とそれを回避する方法を理解するのを助けることができますか?

+1

あなたの答えが見つかったようですが、もう1つ気がついたことです。ベースクラスとサブクラスを両方とも '@ Entity'クラスにしたいのですか?それらはデータストアに別々の種類として格納されます。本当のポリモプリズムと型を問う能力が必要な場合は、サブクラスで '@ Subclass'を使うべきです。 – stickfigure

+0

@stickfigure:そうです。事は、現在、私はちょうどデータベースにエンティティを何とか永続させることに焦点を合わせていて、Genericlearnerがそれの1つであることが判明したということです。私は、後で、一般的な学習者をStudentなどに拡張し、その時点で基本クラスからエンティティアノテーションを削除することにします。あなたは当面、テスト目的のためだと言うことができます。 –

答えて

2

問題は、IDSは彼らがそうではなく、埋め込まれたエンティティのためにautogenerated-されている期待しています。 出典:だからhttps://groups.google.com/forum/#!topic/objectify-appengine/TC2G5I-GJ8w

@stickfigureがhereを述べたように:

あなたは2つのオプションがあります。

  1. はあなたの埋め込みオブジェクトにIDを与えてはいけません。 @Entityを与えず、idフィールドを与えないでください(少なくとも@Idを削除してください)。ちょうど POJOです。 90%の時間で、これは人々が埋め込みオブジェクトで望むものです。
  2. あなた自身(デフォルトではない)のコンストラクタ内に、アロケータで自分自身を割り当てます。

そして、

私はさらに一歩行くと、これまで任意のエンティティのID自動生成 を使用することはありませんお勧めします。エンティティの(デフォルトではない) コンストラクタで常にアロケータを使用してください。これにより、エンティティは常に有効で安定したIDの になります。 トランザクションが開始される前に常にidを割り当てると、 トランザクションが再試行されたときに作成される重複したエンティティが修正されます。ヌルIDを取り込むことはまったく悪い考えです。 の周りには本当にGAEに追加してはいけません。 http://code.google.com/p/objectify-appengine/wiki/IntroductionToObjectify#Entity_Representation から

また、ライリー氏、

、それは実際にすべてのデータストアで エンティティを分離している@Embedded実体のようには見えません。それらは単に親オブジェクトのフィールドに直接 のプロパティとして格納されます。

子エンティティを個別にアクセスできる オブジェクトにする場合は、ParentにキーまたはIDの配列を保持するように変更し、 は子オブジェクトを別々にインスタンス化する必要があります。これにより、各子(親のid配列に格納できる)のID番号 が生成されます。

埋め込みオブジェクトを非エンティティにすることで私の問題を解決しました。したがって、それらのIDを必要としません。@Entityアノテーションを削除しましたが、今はスムーズに動作しています。

関連する問題