spring - JpaRepository 与不同表上的本机 @Query
问题描述
我只是偶然发现了春季数据中的一些意外行为。为了演示,我使用 Initializr ( https://start.spring.io/ ) 添加了 JPA、Web、H2设置了一些 Spring Boot 应用程序。
该应用程序包含两个表和一些数据:
数据.sql:
create table table1 (
id int,
name varchar(50)
);
create table table2 (
id int,
name varchar(50)
);
insert into table1 (id, name) values (1, 'First row from table 1');
insert into table1 (id, name) values (2, 'Second row from table 1');
insert into table1 (id, name) values (3, 'Third row from table 1');
insert into table1 (id, name) values (4, 'Fourth row from table 1');
insert into table2 (id, name) values (1, '** TABLE 2: 1st ROW UPPERCASE **');
insert into table2 (id, name) values (2, '** TABLE 2: 2nd ROW UPPERCASE **');
此表结构只有一个模型,因为两个表的结构相同。我为这个表创建了一个 JpaRepository
示例DAO:
@Repository
public interface SampleDAO extends JpaRepository<SampleModel, Integer> {
@Query(value = "select id, name from table1", nativeQuery = true)
List<SampleModel> findAllFromTable1();
@Query(value = "select id, name from table2", nativeQuery = true)
List<SampleModel> findAllFromTable2();
}
最后我添加了一个控制器(TestController):
@RestController
public class TestController {
@Autowired
private final SampleDAO sampleDAO;
public TestController(SampleDAO sampleDAO) {
this.sampleDAO = sampleDAO;
}
@GetMapping(path = "/")
@ResponseBody
public String testNativeQuery() {
List<SampleModel> list1 = this.sampleDAO.findAllFromTable1();
List<SampleModel> list2 = this.sampleDAO.findAllFromTable2();
SampleModel m1 = list1.get(0);
SampleModel m2 = list2.get(0);
System.out.println("*****************************************");
System.out.println("Data from findAllFromTable1():");
list1.forEach(l -> {
System.out.println(l.getName());
});
System.out.println("*****************************************");
System.out.println("Data from findAllFromTable2():");
list2.forEach(l -> {
System.out.println(l.getName());
});
System.out.println("*****************************************");
return "Done";
}
}
我希望 list1 和 list2 包含我的两个表的数据,但令人惊讶的是,只获取了第一个表的结果:
*****************************************
Data from findAllFromTable1():
First row from table 1
Second row from table 1
Third row from table 1
Fourth row from table 1
*****************************************
Data from findAllFromTable2():
First row from table 1
Second row from table 1
*****************************************
您可以在此处找到示例项目:https ://github.com/steinmann321/nativequerydemo
这是预期的行为还是我做错了什么?
(请注意:这只是一个示例项目,现实生活中的查询要复杂得多,但结果是一样的)
解决方案
您在持久实体之间存在身份冲突问题。您可以看到您从第一个查询中获得了四条记录,从第二个查询中获得了两条与数据库中的记录匹配的记录。但由于实体管理器正在处理来自第二组的记录,它看到已经存在 ID 为 1 和 2 的持久实体,因此它不会创建新实体。
如果你想让它工作,你需要表明你有一个由 id 和 name 组成的复合键。这也将匹配您的 equals 方法中的业务逻辑。
创建 JPA 可嵌入密钥:
@Embeddable
public class MyKey implements Serializable {
private static final long serialVersionUID = 1L;
@Column(name = "id", nullable = false)
private String id;
@Column(name = "name", nullable = false)
private String name;
public String getId() {
return id;
}
public void setId(String id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@Override
public boolean equals(Object o) {
if (this == o) {
return true;
}
if (o == null || getClass() != o.getClass()) {
return false;
}
MyKey that = (MyKey) o;
return Objects.equals(id, that.getId()) && Objects.equals(name, that.getName());
}
@Override
public int hashCode() {
return Objects.hash(id, name);
}
}
然后使用 SampleModel 实体中的嵌入键:
@Entity
public class SampleModel {
@EmbeddedId
private MyKey myKey;
public MyKey getMyKey() {
return myKey;
}
public void setMyKey(MyKey myKey) {
this.myKey = myKey;
}
}
您的控制器将稍作更改以从键中提取值。
System.out.println("*****************************************");
System.out.println("Data from findAllFromTable1():");
list1.forEach(l -> {
System.out.println(l.getMyKey().getName());
});
System.out.println("*****************************************");
System.out.println("Data from findAllFromTable2():");
list2.forEach(l -> {
System.out.println(l.getMyKey().getName());
});
System.out.println("*****************************************");
然后试试你的 TestController。我确实尝试过,结果如下:
*****************************************
Data from findAllFromTable1():
First row from table 1
Second row from table 1
Third row from table 1
Fourth row from table 1
*****************************************
Data from findAllFromTable2():
** TABLE 2: 1st ROW UPPERCASE **
** TABLE 2: 2nd ROW UPPERCASE **
*****************************************
推荐阅读
- spring - 引起:java.lang.AbstractMethodError
- javascript - OpenLayers 的 2D 画布库。绘制自定义交互对象
- r - 如何在保留 (") 的同时使用 (\") 从字符串中删除 (\)?
- r - 从特定的间隔序列创建向量
- jsf - 验证失败时保持 p:dialog 打开
- python - 合并包含 COCO 人物关键点注释的 json 文件
- angular - 不断变化的数据中的按钮单击问题
- python - 如何在 selenium python 或使用 selenium 库的机器人框架中使用鼠标放大网页?
- nginx - NGINX 没有路由到 net core 项目(端口不是 80)
- python - 如何在python中将字符串值调用到sql查询中