我有一个问题要问:
我必须通过user.id将表格user和user_login加入OneToOne - > user_login.user_id。
问题是当我这样做.updateObject(用户)我执行了2次查询:
Hibernate:插入用户(创建,修改,电子邮件,first_name, last_name)values(?,?,?,?,?)Hibernate:插入user_login (创建,修改,密码,user_id)值(?,?,?,?)[2012-08-15 12:15:04,192] [ERROR] [http-bio-8080-exec-1] SqlExceptionHelper [144]: 列'user_id'不能为空
看起来两个对象之间没有引用。如果进入实体用户,方法setUserLogin我添加行 userLogin.setUser(本);它的工作,但我没有找到这种方式优雅诚实。我在实体配置中遗漏了什么吗? 也许这不会自动完成?
谢谢
以下是我的实体
@Entity
@NamedQueries({ @NamedQuery(name = "user.list", query = "select u from User u") })
public class User implements java.io.Serializable {
@Column(name = "first_name", nullable = true)
private String firstName;
@OneToOne(mappedBy = "user", cascade = CascadeType.ALL, fetch = FetchType.LAZY)
@JoinColumn(name="user_id", nullable=false)
private UserLogin userLogin;
public String getFirstName() {
return firstName;
}
public void setFirstName(String firstName) {
this.firstName = firstName;
}
public UserLogin getUserLogin() {
return userLogin;
}
public void setUserLogin(UserLogin userLogin) {
this.userLogin = userLogin;
//userLogin.setUser(this); THIS IS THE LINE THAT FIXES IT, BUT I DONT FIND THIS WAY ELEGANT
}
}
@Entity
@Table(name="user_login")
public class UserLogin implements java.io.Serializable {
@Column(name = "password", nullable = false)
private String password;
@OneToOne(optional = false, fetch = FetchType.LAZY)
private User user;
public String getPassword() {
return password;
}
public void setPassword(String password) {
this.password = password;
}
public User getUser() {
return user;
}
public void setUser(User user) {
this.user = user;
}
}
JSP文件:
<%@ taglib uri="http://www.springframework.org/tags/form" prefix="form"%>
<%@ taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c"%>
<%@ taglib uri="http://java.sun.com/jsp/jstl/functions" prefix="fn"%>
<html>
<head>
<title>Registration Page</title>
</head>
<body>
<form:form action="/test" commandName="user">
<tr>
<td>User Name :</td>
<td><form:input path="firstName" /></td>
</tr>
<tr>
<td>Password :</td>
<td><form:input path="userLogin.password" /></td>
</tr>
<tr>
<td colspan="2"><input type="submit" value="Register"></td>
</tr>
</table>
</form:form>
</body>
</html>
Spring Controller:
@Controller(value = "/")
public class Test {
@Autowired
UserServiceImpl userServiceImpl;
@RequestMapping(method = RequestMethod.GET, value = "/test")
public void test(ModelMap model) {
model.addAttribute("user", new User());
}
@RequestMapping(method = RequestMethod.POST, value = "/test")
public void test(User user) {
userServiceImpl.update(user);
}
}
答案 0 :(得分:2)
与往常一样,双向关系确实拥有自己的一面。拥有关系的一方是mappedBy引用的属性。在您的情况下,user
实体中的属性UserLogin
是拥有方。
当关系持久保存到数据库时,只咨询拥有方。这意味着,您必须设置user
属性的值才能持久化。为了使实体图在内存中保持一致,应该设置关系的两个边。
在JPA 2.0规范中,用以下词语告知:
托管实体之间的双向关系将保持不变 基于关系所属方的参考。它是 开发人员有责任保留内存中的引用 在拥有方和那些与反方保持一致的方面 彼此改变时彼此。