0

テクノロジー:Spring、Hibernate、JSR-303、JQuery
プラットフォーム:Windows
@IdMustExistJSR-303制約を実装しようとしています。制約の目的は、入力されたID値が関連するテーブルに存在するかどうかを確認することです。IdMustExist.javaおよびIdMustExistValidator.javaコードスニペットについては、このリンクを参照してください。

シナリオ1-有効なID値が入力されました:この場合、hibernateのEntityManager.merge操作が実行されると、@IdMustExist制約が実行されていることがわかります。入力されたID値が別の関連付けられた/リンクされたテーブルに存在することを正常に検証します。Hibernateは保存操作を正常に完了します。@IdMustExistのバリデーターを呼び出す前に、hibernateがエンティティのselectSQLを起動することに気付きました。

シナリオ2-無効なID値が入力されました:この場合、hibernateのEntityManager.merge操作が実行されると、入力された無効なIDのエンティティが見つからなかったため、EntityNotFoundException(以下に示すスタックトレース)がスローされます。制約@IdMustExistが更新前の段階で発生し、ユーザーにエラーメッセージを適切に表示できることを期待していました。しかし、休止状態が更新前の段階に入る前に何かが失敗したようです。hibernateがエンティティのselectSQLを起動してから、EntityNotFoundExceptionをスローすることがわかります。validate@IdMustExistを呼び出す機会はありません。

入力したIDがJSR-303を介して関連付けられた/リンクされたテーブルに存在するかどうかを検証できないということですか?代替案はありますか?

よろしくお願いします。JP

javax.persistence.EntityNotFoundException: Unable to find com.mycompany.myapp.domain.package1.Class1 with id 100
at org.hibernate.ejb.Ejb3Configuration$Ejb3EntityNotFoundDelegate.handleEntityNotFound(Ejb3Configuration.java:133)
at org.hibernate.event.def.DefaultLoadEventListener.load(DefaultLoadEventListener.java:233)
at org.hibernate.event.def.DefaultLoadEventListener.proxyOrLoad(DefaultLoadEventListener.java:285)
at org.hibernate.event.def.DefaultLoadEventListener.onLoad(DefaultLoadEventListener.java:152)
at org.hibernate.impl.SessionImpl.fireLoad(SessionImpl.java:1080)
at org.hibernate.impl.SessionImpl.internalLoad(SessionImpl.java:1028)
at org.hibernate.type.EntityType.resolveIdentifier(EntityType.java:623)
at org.hibernate.type.EntityType.resolve(EntityType.java:431)
at org.hibernate.type.EntityType.replace(EntityType.java:291)
at org.hibernate.type.TypeFactory.replace(TypeFactory.java:532)
at org.hibernate.event.def.DefaultMergeEventListener.copyValues(DefaultMergeEventListener.java:495)
at org.hibernate.event.def.DefaultMergeEventListener.entityIsDetached(DefaultMergeEventListener.java:423)
at org.hibernate.event.def.DefaultMergeEventListener.onMerge(DefaultMergeEventListener.java:234)
at org.hibernate.event.def.DefaultMergeEventListener.onMerge(DefaultMergeEventListener.java:84)
at org.hibernate.impl.SessionImpl.fireMerge(SessionImpl.java:859)
at org.hibernate.impl.SessionImpl.merge(SessionImpl.java:843)
at org.hibernate.impl.SessionImpl.merge(SessionImpl.java:847)
at org.hibernate.ejb.AbstractEntityManagerImpl.merge(AbstractEntityManagerImpl.java:682)
at com.mycompany.myapp.dao.hibernate.package2.HibernateClass2Dao.save(HibernateClass2Dao.java:101)
at com.mycompany.myapp.service.package2.Class1ServiceImpl.update(Class1ServiceImpl.java:56)
    ...
    ...
at java.lang.Thread.run(Thread.java:619)

HibernateClass2Dao.java

    public void save(GenericRequest request, GenericResponse response){
    Class2 class2Request = (Class2) request.getObject(Class2.class.getName());
    EntityManager em = PersistenceUtil.getEntityManagerFactory().createEntityManager();
    Class2 class2Persisted = em.find(Class2.class, class2Request.getId());
    EntityTransaction tx = em.getTransaction();
    try {
        tx.begin();
        class2Persisted.setClass1(class2Request.getClass1());
        em.merge(class2Request);
        tx.commit();            
    } catch (RollbackException rbe) {
        response.setSuccess(false);
        rbe.getCause().printStackTrace();
        ConstraintViolationException cve = (ConstraintViolationException) rbe.getCause();
        Set<ConstraintViolation<?>> constraintViolations = cve.getConstraintViolations();
        if (constraintViolations.size() > 0){
            for (ConstraintViolation<?> violation : constraintViolations){
                Iterator<Node> itr = violation.getPropertyPath().iterator();
                String propertyPath = itr.next().getName();
                Class<? extends Payload> payload = violation.getConstraintDescriptor().getPayload().iterator().next();
                String payloadName = payload.getCanonicalName();                
                response.addMessage(violation.getMessage(), violation.getLeafBean().getClass().getName(),
                        propertyPath , payloadName);            
            }
        }
    } finally {
        em.close();         
    }
}

Class2ServiceImpl.java

    public void update(GenericRequest request, GenericResponse response){
    class2Dao.save(request, response);
}

Class2Controller.java

    @RequestMapping(value="/update")
public @ResponseBody GenericResponse update(String jqGridId, String oper
            , Class2 class2
            , BindingResult bindingResult) throws Exception {
    GenericResponse response = new GenericResponse(true);
    response.setMessageSource(messageSource);
    Locale locale = new Locale(CommonConstants.DEFAULT_LOCALE);
    if (bindingResult.hasErrors()){
        response.setSuccess(false);
        addBindingMessages(response, bindingResult, locale);
        return response;
    }
    class2.setId(Long.parseLong(jqGridId));
    class2.setCode(jqGridId);
    GenericRequest request = new GenericRequest(UserUtil.getUser(), class2);
    this.class2Service.update(request, response);
    return response;        
}   
4

2 に答える 2

1

参照されているリンクにはDAOのコードが表示されていませんがload()、セッションのメソッドを使用していると思います。javadocを参照してください。

インスタンスが存在すると仮定して、指定された識別子を持つ指定されたエンティティクラスの永続インスタンスを返します。

「インスタンスが存在すると仮定して」と書かれている部分に注意してください。つまり、いくつかのエンティティが存在することを知っており、Hibernateにそれをロードするように要求しています。存在しない場合、この仮定は破られ、例外がスローされます。

get()代わりにメソッドを使用することをお勧めします。javadocの内容を参照してください。

指定された識別子を持つ指定されたエンティティクラスの永続インスタンスを返します。そのような永続インスタンスがない場合はnullを返します。

したがって、解決策はからに変更するload()ことget()です。

Javadoc:http ://docs.jboss.org/hibernate/core/3.5/api/org/hibernate/Session.html

于 2011-02-02T07:45:53.567 に答える
0

ここにいくつかのことがあります。CommonDaoが使用しているセッションは何ですか。実際、これは現在の接続で開かれた新しいセッションである必要があります。セッション(または制約バリデーター内のエンティティマネージャー)の使用に関する問題全体は、Hibernate Validator wikiで説明されています:http://community.jboss.org/wiki/AccessingtheHibernateSessionwithinaConstraintValidator

しかし、私はこのアプローチをお勧めしません。マージを呼び出して、Hibernate Core(またはJPA)の例外を処理します。

于 2011-02-05T15:07:58.140 に答える