1

で注釈が付けられたフィールドを持つJavaクラスがあります@SearchCriteria(criteria = "class1.class2.field")criteria注釈内のパラメータは、クラスを使用する場合、このフィールドを休止状態の基準として設定する必要があることを意味します。これは、たとえば、フィールドが次のようにマークされている場合を意味します。

@SearchCriteria(criteria = "class1.class2.field")
private String value;

次のような休止状態の基準を動的に作成したい

DetachedCriteria hibernateCriteria = forClass(Class.class);
hibernateCriteria.createCriteria("class1").createCriteria("class2").add(eq("field", value));

問題は、すでに追加されている別の基準を設定できないことです。つまり、注釈criteriaオプションを確認する必要があります。

      switch (annotationCriteria.length - 1) {
case 0:             
hibernateCriteria.add(isNull(annotationCriteria[0]));               
case 1:
hibernateCriteria.createCriteria(annotationCriteria[0]).add(                Restrictions.isNull(annotationCriteria[annotationCriteria.length - 1]));
case 2:   hibernateCriteria.createCriteria(annotationCriteria[0]).createCriteria(annotationCriteria[1]).add(                Restrictions.isNull(annotationCriteria[annotationCriteria.length - 1]));
    }

この「swith」を削除したいと思います。たとえば「名前」によって、すでに追加された基準を取得することは可能ですが、新しい副基準を追加することはできません。

4

2 に答える 2

1

私はこの問題のより「良い」解決策を見つけました。まず、searchCriteriaクラスを作成します。次に例を示します。

public class SearchCriteria {

    @SearchCriteria(path = "someclass.someclass.id")
    private Integer someId;

    public Integer getSomeId() {
        return someId;
    }

    public void setSomeId(Integer someId) {
        this.someId= someId;
    }
}

そしてカスタムアノテーション:

@Target(value = { ElementType.FIELD })
@Retention(RUNTIME)
public @interface SearchCriteria {
    String path() default "";

    boolean optional() default true;
}

そしてDAOクラスでは:

@Override
    public Collection<SomeClass> find(SearchCriteria criteria) {
        DetachedCriteria hibernateCriteria = forClass(SomeClass.class);

        for (Field field : criteria.getClass().getDeclaredFields()) {
            Annotation annotation = field.getAnnotation(SearchCriteria.class);

            if (annotation == null) {
                continue;
            }

            List<String> elements = Arrays.asList(StringUtils.split(((SearchCriteria) annotation).path(), "."));

            field.setAccessible(true);
            Object fieldValue = null;
            try {
                fieldValue = field.get(criteria);
            } catch (IllegalArgumentException e) {
                LOG.error("while trying to get private field value of entity " + SearchCriteria.class, e);
            } catch (IllegalAccessException e) {
                LOG.error("while trying to get private field value of entity " + SearchCriteria.class, e);
            }

            for (String element : elements) {
                if (elements.indexOf(element) == elements.size() - 1) {
                    hibernateCriteria = hibernateCriteria.add(eq(element, fieldValue));
                } else {
                    hibernateCriteria = hibernateCriteria.createCriteria(element);
                }
            }
        }

        return getHibernateTemplate().findByCriteria(hibernateCriteria);
    }
于 2012-01-20T15:11:35.123 に答える
1

いいえ、できません。Detached Criteria APIは、良くも悪くも、検出を許可しないため、「既存の」基準を要求することはできません。

ただし、できることは、関連付けパスごとにネストされた基準の独自のマップを維持することです。擬似コードの場合:

Map<String, DetachedCriteria> criteriaMap = ...;

for ( ) { // loop over annotation criteria's "elements"
    DetachedCriteria existing = criteriaMap.get(fullPath);
    if (existing==null) {
        existing = parentCriteria.createCriteria(pathElement);
        criteriaMap.put(fullPath, existing);
    }
    existing.add(whateverCondition);
}
于 2011-09-23T19:05:53.007 に答える