如何将构造函数映射与 Spring JPA 存储库一起使用
How to use Constructor Mapping with Spring JPA Repositories
我有一个 Spring 存储库,如下所示:
import org.springframework.data.repository.Repository;
import org.springframework.stereotype.Component;
import com.test.domain.My;
@Component
public interface MyRepository extends Repository<My, String> {
My findOne(String code);
My findByName(String name);
}
实体class是:
import javax.persistence.ColumnResult;
import javax.persistence.ConstructorResult;
import javax.persistence.Entity;
import javax.persistence.Id;
import javax.persistence.SqlResultSetMapping;
import javax.persistence.Table;
import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonIgnoreProperties;
import com.fasterxml.jackson.annotation.JsonProperty;
@JsonIgnoreProperties(ignoreUnknown=true)
@Entity
@Table(name="vMy", schema="test")
@SqlResultSetMapping(
name="something",
classes = {
@ConstructorResult(targetClass = My.class,
columns={
@ColumnResult(name = "myCode", type = String.class),
@ColumnResult(name = "myShortName", type = String.class)
}
)
}
)
public class My {
@Id
@Column(name = "myCode")
private final String code;
@Column(name = "myShortName")
private final String name;
public My(String code, String name) {
this.code = code;
this.name = name;
}
@JsonCreator()
public My(@JsonProperty("My_c") String code) {
this.code = code;
this.name = null;
}
public String getCode() {
return code;
}
public String getName() {
return name;
}
@Override
public String toString() {
return "{code: " + code + ", name: " + name + "}";
}
}
调用findOne或findByName时报错如下:
org.hibernate.InstantiationException: No default constructor for entity
如何在没有默认构造函数的情况下使用 Spring JPA 存储库?我想保留实例字段、代码和名称,最终。
我会创建一个名为 MyDto
的单独 class,它具有 JSON 内容,但没有实体注释。使其字段也成为最终字段。
那么你的存储库方法将是这样的:
@Query("SELECT new MyDto(m.code, m.name) FROM My m WHERE m.code = :code")
public MyDto findByCode(@Param("code") String code);
这样,您仅使用我的实体 class 来为您提供到数据库列的映射,而不是创建 My
.
的实例
编辑:另一种方法(as detailed here)是使用实体class本身作为DTO。
因此您的查询方法可能如下所示:
@Query("SELECT new My(m.code, m.name) FROM My m WHERE m.code = :code")
public My findByCode(@Param("code") String code);
这样做的好处是不必创建单独的 DTO class。
我有一个 Spring 存储库,如下所示:
import org.springframework.data.repository.Repository;
import org.springframework.stereotype.Component;
import com.test.domain.My;
@Component
public interface MyRepository extends Repository<My, String> {
My findOne(String code);
My findByName(String name);
}
实体class是:
import javax.persistence.ColumnResult;
import javax.persistence.ConstructorResult;
import javax.persistence.Entity;
import javax.persistence.Id;
import javax.persistence.SqlResultSetMapping;
import javax.persistence.Table;
import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonIgnoreProperties;
import com.fasterxml.jackson.annotation.JsonProperty;
@JsonIgnoreProperties(ignoreUnknown=true)
@Entity
@Table(name="vMy", schema="test")
@SqlResultSetMapping(
name="something",
classes = {
@ConstructorResult(targetClass = My.class,
columns={
@ColumnResult(name = "myCode", type = String.class),
@ColumnResult(name = "myShortName", type = String.class)
}
)
}
)
public class My {
@Id
@Column(name = "myCode")
private final String code;
@Column(name = "myShortName")
private final String name;
public My(String code, String name) {
this.code = code;
this.name = name;
}
@JsonCreator()
public My(@JsonProperty("My_c") String code) {
this.code = code;
this.name = null;
}
public String getCode() {
return code;
}
public String getName() {
return name;
}
@Override
public String toString() {
return "{code: " + code + ", name: " + name + "}";
}
}
调用findOne或findByName时报错如下:
org.hibernate.InstantiationException: No default constructor for entity
如何在没有默认构造函数的情况下使用 Spring JPA 存储库?我想保留实例字段、代码和名称,最终。
我会创建一个名为 MyDto
的单独 class,它具有 JSON 内容,但没有实体注释。使其字段也成为最终字段。
那么你的存储库方法将是这样的:
@Query("SELECT new MyDto(m.code, m.name) FROM My m WHERE m.code = :code")
public MyDto findByCode(@Param("code") String code);
这样,您仅使用我的实体 class 来为您提供到数据库列的映射,而不是创建 My
.
编辑:另一种方法(as detailed here)是使用实体class本身作为DTO。
因此您的查询方法可能如下所示:
@Query("SELECT new My(m.code, m.name) FROM My m WHERE m.code = :code")
public My findByCode(@Param("code") String code);
这样做的好处是不必创建单独的 DTO class。