Спецификация Spring Data JPA с использованием CriteriaBuilder с отношением один ко многим
у меня есть User
сущность, a UserToApplication
сущность, и Application
сущности.
один User
может иметь доступ к более чем один Application
. И один Application
может использоваться более чем одним User
.
здесь User
сущности.
@Entity
@Table(name = "USER", schema = "UDB")
public class User {
private Long userId;
private Collection<Application> applications;
private String firstNm;
private String lastNm;
private String email;
@SequenceGenerator(name = "generator", sequenceName = "UDB.USER_SEQ", initialValue = 1, allocationSize = 1)
@Id
@GeneratedValue(strategy = GenerationType.SEQUENCE, generator = "generator")
@Column(name = "USER_ID", unique = true, nullable = false)
public Long getUserId() {
return userId;
}
public void setUserId(Long userId) {
this.userId = userId;
}
@OneToMany(mappedBy = "user", fetch = FetchType.LAZY)
public Collection<Application> getApplications() {
return applications;
}
public void setApplications(Collection<Application> applications) {
this.applications = applications;
}
/* Other getters and setters omitted for brevity */
}
здесь UserToApplication
сущности.
@Entity
@Table(name = "USER_TO_APPLICATION", schema = "UDB")
public class Application {
private Long userToApplicationId;
private User user;
private Application application;
@SequenceGenerator(name = "generator", sequenceName = "UDB.USER_TO_APP_SEQ", initialValue = 0, allocationSize = 1)
@Id
@GeneratedValue(strategy = GenerationType.SEQUENCE, generator = "generator")
@Column(name = "USER_TO_APPLICATION_ID", unique = true, nullable = false)
public Long getUserToApplicationId() {
return userToApplicationId;
}
public void setUserToApplicationId(Long userToApplicationId) {
this.userToApplicationId = userToApplicationId;
}
@ManyToOne
@JoinColumn(name = "USER_ID", referencedColumnName = "USER_ID", nullable = false)
public User getUser() {
return user;
}
public void setUser(User user) {
this.user = user;
}
@ManyToOne
@JoinColumn(name = "APPLICATION_ID", nullable = false)
public Application getApplication() {
return application;
}
}
и вот это Application
сущности.
@Entity
@Table(name = "APPLICATION", schema = "UDB")
public class Application {
private Long applicationId;
private String name;
private String code;
/* Getters and setters omitted for brevity */
}
у меня есть следующие Specification
что я использую для поиска User
by firstNm
, lastNm
и email
.
public class UserSpecification {
public static Specification<User> findByFirstNmLastNmEmail(String firstNm, String lastNm, String email) {
return new Specification<User>() {
@Override
public Predicate toPredicate(Root<User> root, CriteriaQuery<?> query, CriteriaBuilder cb) {
final Predicate firstNmPredicate = null;
final Predicate lastNmPredicate = null;
final Predicate emailPredicate = null;
if (!StringUtils.isEmpty(firstNm)) {
firstNmPredicate = cb.like(cb.lower(root.get(User_.firstNm), firstNm));
}
if (!StringUtils.isEmpty(lastNm)) {
lastNmPredicate = cb.like(cb.lower(root.get(User_.lastNm), lastNm));
}
if (!StringUtils.isEmpty(email)) {
emailPredicate = cb.like(cb.lower(root.get(User_.email), email));
}
return cb.and(firstNmPredicate, lastNmPredicate, emailPredicate);
}
};
}
}
и вот это User_
метамодель, которая у меня есть до сих пор.
@StaticMetamodel(User.class)
public class User_ {
public static volatile SingularAttribute<User, String> firstNm;
public static volatile SingularAttribute<User, String> lastNm;
public static volatile SingularAttribute<User, String> email;
}
теперь, я хотел бы также передать список идентификаторов применения к Specification
, так что его сигнатура метода будет:
public static Specification<User> findByFirstNmLastNmEmailApp(String firstNm, String lastNm, String email, Collection<Long> appIds)
Итак, мой вопрос в том, Могу ли я добавить @OneToMany
отображение в User_
метамодель для
1 ответов
я нашел решение. Чтобы сопоставить атрибут один ко многим, в метамодели я добавил следующее:
public static volatile CollectionAttribute<User, Application> applications;
мне также нужно было добавить метамодель для Application
сущности.
@StaticMetamodel(Application.class)
public class Application_ {
public static volatile SingularAttribute<Application, Long> applicationId;
}
тогда в моем Specification
, я мог бы получить доступ к applications
для пользователя, используя .join()
метод Root<User>
экземпляра. Вот это Predicate
Я создал.
final Predicate appPredicate = root.join(User_.applications).get(Application_.applicationId).in(appIds);
кроме того, стоит отметить, что мой Specification
как написано в вопросе, не будет работать, если любой из входные значения пусты. A null Predicate
перешло к .and()
метод CriteriaBuilder
вызывает NullPointerException
. Итак, я создал ArrayList
типа Predicate
, затем добавляют каждый Predicate
в список, если соответствующий параметр не пуст. Наконец, я конвертирую ArrayList
в массив, чтобы передать его в