Spring Data JPA-Spezifikation mit CriteriaBuilder mit ein eins-zu-viele-Beziehung
Ich habe eine User
Einheit, eine UserToApplication
Einheit, und eine Application
Einheit.
Einem einzigen User
können haben Zugriff auf mehr als eine Application
. Und eine einzelne Application
verwendet werden kann, die in mehr als einem User
.
Hier ist die User
Einheit.
@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 */
}
Hier ist die UserToApplication
Einheit.
@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;
}
}
Und hier ist die Application
Einheit.
@Entity
@Table(name = "APPLICATION", schema = "UDB")
public class Application {
private Long applicationId;
private String name;
private String code;
/* Getters and setters omitted for brevity */
}
Habe ich Folgendes Specification
dass ich auf der Suche nach einem User
durch firstNm
, lastNm
, und 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);
}
};
}
}
Und hier ist die User_
- Metamodell, dass ich so weit.
@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;
}
Nun, ich möchte auch in eine Liste mit Anwendungs-IDs, um die Specification
, so dass seine Methode Signatur wäre:
public static Specification<User> findByFirstNmLastNmEmailApp(String firstNm, String lastNm, String email, Collection<Long> appIds)
So, meine Frage ist, wenn ich die @OneToMany
Zuordnung zu den User_
Metamodell für die Collection<Application> applications
Bereich meiner User
Person, wie könnte ich dann Verweis in der Specification
?
Meine aktuelle Specification
würde etwa die folgende SQL-Abfrage:
select * from user u
where lower(first_nm) like '%firstNm%'
and lower(last_nm) like '%lastNm%'
and lower(email) like '%email%';
Ist und was ich erreichen möchte in der neuen Specification
wäre so etwas wie dieses:
select * from user u
join user_to_application uta on uta.user_id = u.user_id
where lower(u.first_nm) like '%firstNm%'
and lower(u.last_nm) like '%lastNm%'
and lower(u.email) like '%email%'
and uta.application_id in (appIds);
Ist es möglich, das zu tun diese Art der Zuordnung in das Metamodell, und wie könnte ich dieses Ergebnis in mein Specification
?
Du musst angemeldet sein, um einen Kommentar abzugeben.
Habe eine Lösung gefunden. Auf Karte eins-zu-viele-Attribut, in das Metamodell habe ich noch folgende:
Brauchte ich auch hinzufügen, ein Metamodell für die
Application
Einheit.Dann in meinem
Specification
, konnte ich Zugriff auf dieapplications
für einen Benutzer, mit dem.join()
Methode auf dieRoot<User>
Instanz. Hier ist diePredicate
ich gebildet.Es ist auch erwähnenswert, dass meine
Specification
wie es geschrieben steht in der Frage wird nicht funktionieren, wenn einer der Eingabewerte leer sind. Ein null -Predicate
an die.and()
Methode derCriteriaBuilder
verursachenNullPointerException
. So, ich habe eineArrayList
TypPredicate
, dann hat jedesPredicate
auf der Liste, wenn der entsprechende parameter nicht leer war. Endlich, ich konvertiere dieArrayList
um ein array zu übergeben es an die.and()
Funktion derCriteriaBuilder
. Hier ist der LetzteSpecification
: