У меня есть веб-сайт стороннего проекта, где люди могут комментировать вещи, теперь мне нужно отображать комментарии пользователей и имя пользователя комментатора. Проблема в том, что я не могу получить имя пользователя, вошедшего в систему.
Примеры комментариев:
1)
Прокомментировал - Шело
Комментарий к тесту
2)
Прокомментировал - ProGa
Комментарий к тесту 2
Теперь проблема в том, что если я попытаюсь получить имя пользователя «в данный момент вошедшего в систему», оба комментария будут иметь одно и то же имя пользователя, когда пользователь войдет в систему.
Authentication loggedInUser = SecurityContextHolder.getContext().getAuthentication();
String name = loggedInUser.getName();
В приведенном выше примере кода показано, как вы получаете пользователя, вошедшего в систему. Итак, если я использую этот код и попытаюсь отобразить имя пользователя с помощью String name, и, скажем, какой-то пользователь войдет в систему с именем пользователя — "Давос", оба комментария будут выглядеть так:
1)
Прокомментировал - Давос
Комментарий к тесту
2)
Прокомментировал - Давос
Комментарий к тесту 2
Что мне нужно, так это получить имя пользователя, который вошел в систему, и если он оставит комментарий, отобразить его имя пользователя. Когда он выходит из системы или какой-либо другой пользователь входит в систему, «имя пользователя» комментария должно оставаться прежним и не должно меняться в зависимости от того, кто в данный момент вошел в систему.
Controller Class (only relevant code):
@GetMapping("/foodDescription/{id}")
public String foodDescriptionPage(Model model, @PathVariable(name = "id") String id){
menuRepository.findById(id).ifPresent(o -> model.addAttribute("menu", o));
return "food-description";
}
@PostMapping("/postComment/{id}")
public String postComment(@RequestParam String myComment, Model model, @PathVariable(name = "id") String id){
Optional<Menu> menu = menuRepository.findById(id);
menuRepository.findById(id).ifPresent(o -> model.addAttribute("menu", o));
List<Comments> myCommentsList = menu.get().getComments();
myCommentsList.add(new Comments(myComment));
menu.get().setComments(myCommentsList);
menuRepository.save(menu.get());
return "food-description";
}
food-description.html (Only relevant code for comments section) :
<h3 th:text = "Comments">asd</h3>
<form th:action = "@{/postComment/{myMenuId}(myMenuId=${menu.id})}" method = "POST">
<textarea rows = "2" cols = "100" name = "myComment"></textarea>
<input type = "submit" th:value = "Add"/>
</form>
<div class = "comment" th:each = "com : ${menu.comments}">
<h3 th:text = "Display Commenter's Username here">Commenter's Username</h3>
<ul>
<li th:text = "${com.comment}">Comment</li>
</ul>
</div>
User entity :
@Entity
@Table(name = "users")
public class User {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
@Column(name = "id")
private Long id;
@Column(name = "username")
private String userName;
@Column(name = "password")
private String password;
@Column(name = "first_name")
private String firstName;
@Column(name = "last_name")
private String lastName;
@Column(name = "email")
private String email;
@OneToOne(cascade = CascadeType.ALL, orphanRemoval = true)
@JoinTable(name = "users_role",
joinColumns = @JoinColumn(name = "user_id"),
inverseJoinColumns = @JoinColumn(name = "role_id"))
private Role role;
@OneToMany(cascade = CascadeType.ALL, orphanRemoval = true)
@JoinTable(name = "users_comments",
joinColumns = @JoinColumn(name = "user_id"),
inverseJoinColumns = @JoinColumn(name = "comment_id"))
private List<Comments> comments = new ArrayList<>();
// Getters/Setters/Constructor
}
Comment entity :
@Entity
@Table(name = "comments")
public class Comments {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private int id;
@Column(name = "comment")
private String comment;
// Getters/Setters/Constructor
То, что сказал Мань, верно. У вас уже есть отношение @OneToMany от User к Comments, все, что вам нужно, это добавить отношение @ManyToOne от Comments к User.




Как упоминалось в комментариях, вы должны изменить свое моделирование данных.
Например, вы можете реализовать что-то вроде этого:
@Entity
public class User {
@OneToMany()
public Collection<Comment> comments;
...
}
@Entity
public class Comment {
@ManyToOne
@JoinColumn(name = "user_fk")
public User user;
...
}
В этом случае у вас есть двунаправленная ассоциация, и вы можете получить комментарий пользователя в коде (и в шаблоне html). Конечно, вы также должны изменить свою модель БД, если вы делаете это вручную.
Поэтому я добавил ассоциацию @ManyToOne, и после создания comment она правильно добавляется в БД. Но когда я пытаюсь получить доступ к user комментария с помощью - comment.getUser().getUserName() - я получаю исключение nullPointerException
И зачем мне менять модель БД?
есть ли столбец user_fk в таблице комментариев в вашей базе данных?
Да . У меня есть ассоциация с User->user_role->roleMenu->menu_ingredient->ingredientsMenu->menu_comments->commentsUser->user_comments->comments
Я думаю, что нашел правильный путь получения имени пользователя, я дам вам знать :)
Хорошо, я это сделал :)) Итак, во-первых, после распечатки всех пользователей с -> system.out.println(comment.getUser()); я обнаружил, что действительно странно, что некоторые выходные данные, где null, а некоторые были нормальными, поэтому я решил, что мне нужно обрезать мои таблицы, потому что null выходные данные были на старых записях (которые были добавлены до создания двунаправленной ассоциации).
После усечения таблиц и создания новых записей я смог вывести username в тимелеафе с помощью <h3 th:text = ${com.user.userName}></h3> . Если у кого-то есть такая же проблема, имейте в виду, что у меня уже есть родительский цикл, где я использую com в качестве переменной, поэтому вы можете использовать только - ${comment.user.userName}, основная вещь заключается в том, что вам нужно пройти через comment, чтобы найти user, а затем перебрать user, чтобы найти поле userName. :)
Это основа моделирования. Вам нужно добавить ссылку на
authorкомментария, добавив свойствоuserIdк сущностиcomment. Затем, когда вам нужно отобразить имя автора, вы запрашиваете его из БД.