admin管理员组文章数量:1345171
I am new to developing REST using CrudRepository. I was wondering the proper way to handle JOINs?
I have three classes, three repo classes and three service classes. Stress, Mood, User. StressRepo, MoodRepo, UserRepo. StressService, MoodService, UserService
An example of my service class:
@Service
public class MoodService {
@Autowired
private MoodRepository moodRepository;
public Mood createMood(Mood mood) {
return moodRepository.save(mood);
}
public Mood getMoodById(Long id) {
return moodRepository.findById(id).orElse(null);
}
public List<Mood> getAllMoods() {
return (List<Mood>) moodRepository.findAll();
}
public void deleteMood(Long id) {
moodRepository.deleteById(id);
}
public Mood updateMood(Long id, Mood updateMood) {
Mood existingMood = moodRepository.findById(id).orElse(null);
if (existingMood != null) {
Mood newMood = new Mood(updateMood.getUserId(), updateMood.getDate(), updateMood.getMood(), updateMood.getDescription());
return moodRepository.save(newMood);
}
return null;
}
}
If I was to do this in SQL to query the DB, I would do something like this..
SELECT * FROM USERS u
LEFT JOIN MOODS m ON u.id = m.USER_ID
LEFT JOIN STRESS s ON u.id = s.USER_ID
WHERE (m.MOOD = 1 OR m.MOOD = 2) AND (m.MOOD = 3 AND (s.STRESS = 4 OR s.STRESS = 5))
I assume I would need a custom controller that would handle this request, but do I need to incorporate the Mood and Stress class inside of the User class?
@Getter
@Setter
@Entity
@NoArgsConstructor
@Table(name = "users")
public class User {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private long id;
@Column(length = 100)
private String name;
@Column(length = 100)
private String userName;
@Column(length = 100)
private String zipCode;
User(String userName, String name, String zipCode) {
this.name = name;
this.userName = userName;
this.zipCode = zipCode;
}
@ManyToMany
Stress stress;
@ManyToMany
Mood mood;
}
I am new to developing REST using CrudRepository. I was wondering the proper way to handle JOINs?
I have three classes, three repo classes and three service classes. Stress, Mood, User. StressRepo, MoodRepo, UserRepo. StressService, MoodService, UserService
An example of my service class:
@Service
public class MoodService {
@Autowired
private MoodRepository moodRepository;
public Mood createMood(Mood mood) {
return moodRepository.save(mood);
}
public Mood getMoodById(Long id) {
return moodRepository.findById(id).orElse(null);
}
public List<Mood> getAllMoods() {
return (List<Mood>) moodRepository.findAll();
}
public void deleteMood(Long id) {
moodRepository.deleteById(id);
}
public Mood updateMood(Long id, Mood updateMood) {
Mood existingMood = moodRepository.findById(id).orElse(null);
if (existingMood != null) {
Mood newMood = new Mood(updateMood.getUserId(), updateMood.getDate(), updateMood.getMood(), updateMood.getDescription());
return moodRepository.save(newMood);
}
return null;
}
}
If I was to do this in SQL to query the DB, I would do something like this..
SELECT * FROM USERS u
LEFT JOIN MOODS m ON u.id = m.USER_ID
LEFT JOIN STRESS s ON u.id = s.USER_ID
WHERE (m.MOOD = 1 OR m.MOOD = 2) AND (m.MOOD = 3 AND (s.STRESS = 4 OR s.STRESS = 5))
I assume I would need a custom controller that would handle this request, but do I need to incorporate the Mood and Stress class inside of the User class?
@Getter
@Setter
@Entity
@NoArgsConstructor
@Table(name = "users")
public class User {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private long id;
@Column(length = 100)
private String name;
@Column(length = 100)
private String userName;
@Column(length = 100)
private String zipCode;
User(String userName, String name, String zipCode) {
this.name = name;
this.userName = userName;
this.zipCode = zipCode;
}
@ManyToMany
Stress stress;
@ManyToMany
Mood mood;
}
Share
Improve this question
asked 2 days ago
letsCodeletsCode
3,0741 gold badge18 silver badges41 bronze badges
2
|
1 Answer
Reset to default 0Your example query implies that Mood
has a user id and that Stress
also has a user id, so I assume that inside your entity models for mood and stress, there is some pointer to a particular user. Or in other words, a given mood or stress row in the database points to only one (not many) users. So your user model should be using @OneToMany
on both mood and stress.
@OneToMany(mappedBy = "user", cascade = CascadeType.ALL, orphanRemoval = true)
private List<Stress> stressList;
@OneToMany(mappedBy = "user", cascade = CascadeType.ALL, orphanRemoval = true)
private List<Mood> moodList;
Conversely your mood and stress models should be using
@ManyToOne(fetch = FetchType.LAZY)
@JoinColumn(name = "user_id")
private User user;
on the user reference.
I assume that is causing a problem for you. That being said, if you want a custom query in a JPA repository, you can just add the query to a text block. For example, in your MoodRepository
you can have something like this:
@Query("""
SELECT DISTINCT u FROM User u
LEFT JOIN u.moodList m
LEFT JOIN u.stressList s
WHERE (m.mood = 1 OR m.mood = 2)
AND (m.mood = 3 AND (s.stress = 4 OR s.stress = 5))""")
List<User> findUsersWithMoodAndStressConditions();
本文标签: javaProper way to handle JOINS with CrudRepository RESTStack Overflow
版权声明:本文标题:java - Proper way to handle JOINS with CrudRepository REST - Stack Overflow 内容由网友自发贡献,该文观点仅代表作者本人, 转载请联系作者并注明出处:http://www.betaflare.com/web/1743809032a2542718.html, 本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容,一经查实,本站将立刻删除。
@OneToMany
and@ManyToMany
annotations (e.g.: baeldung/jpa-hibernate-associations, and baeldung/hibernate-one-to-many). In this case, you can avoid a lot of queries, because ORM will JOIN and map collections for you. You still can use a native query, if performance really matters for you in this project. Check out JPQL and JPA Native Query. – Mykola Commented 2 days agoreturn moodRepository.findById(id).map(existingMood -> { existingMood.setDate(updatedMood.getDate()); // etc. return moodRepository.save(existingMood); }).orElseThrow(() -> new EntityNotFoundException("Mood not found with id " + id));
And... Instead of passing the entireMood
entity object from the controller, you could use aMoodUpdateDto
with only the fields that should be editable. – Mykola Commented 2 days ago