为什么 Arraylist<String> 作为一个长而随机的字符串存储在 H2 数据库中?

Why Arraylist<String> get stored in H2 database as a long and random string?

我在 Spring-Boot (Kotlin) 项目中创建了一个实体(模型或数据 class),其中包含一个类型为 Arraylist 的字段,但是当我在 JSON 来自 Postman 的格式,数组作为一个长的随机字符串存储在数据库中。

当我尝试从数据库中检索数据时,我得到了实际的数组,格式完美。

我的问题是为什么ArrayList会这样存储在H2数据库中???

Evaluation.kt

@Entity
data class Evaluation (
 @Id val id : String,
 val timeStamp : Long,
 val symptoms : ArrayList<String>,
 val travelHistory : Boolean,
 val contactWithCovidPatient : Boolean,
 val evaluatedBy : String,
 var evaluationPercentage : String? = null,
 @ManyToOne var user: User? = null
)

EvaluationController.kt

@RestController
class EvaluationController (val evaluationService: IEvaluationService) {

@PostMapping("evaluate/{userId}")
fun evaluateUser(@PathVariable userId : String, @RequestBody evaluation: Evaluation) : ResponseEntity<Evaluation> =
    ResponseEntity.ok().body(evaluationService.addEvaluation(evaluation, userId))

}

请求正文JSON

{
"id":"e_01",
"timeStamp":"123456789",
"pinCode":"123457",
"travelHistory":true,
"contactWithCovidPatient":true,
"evaluatedBy":"u_01",
"symptoms": ["Fever","Cough"]
}

回应JSON

{
"id": "e_01",
"timeStamp": 123456789,
"symptoms": [
    "Fever",
    "Cough"
],
"travelHistory": true,
"contactWithCovidPatient": true,
"evaluatedBy": "u_01",
"evaluationPercentage": "95",
"user": {
    "id": "u_01",
    "name": "abc01",
    "phoneNumber": "9876543210",
    "pinCode": "123457",
    "covidResult": "Positive"
}
}

H2 数据库Table

这是一个表示序列化 ArrayList 对象的十六进制字符串。有关 Java 中对象序列化的详细信息,请参阅 Serializable Objects

运行 以下代码产生相同的结果:

List<String> symptoms = new ArrayList<>(Arrays.asList("Fever", "Cough"));
ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();
ObjectOutputStream objectOutputStream = new ObjectOutputStream(byteArrayOutputStream);
objectOutputStream.writeObject(symptoms);
byte[] serializedObject = byteArrayOutputStream.toByteArray();
String hex = Hex.encodeHexString(serializedObject); // Apache Commons Codec
System.out.println(hex);
aced0005737200136a6176612e7574696c2e41727261794c6973747881d21d99c7619d03000149000473697a657870000000027704000000027400054665766572740005436f75676878

在数据库中看到的原始字符串是一个序列化对象。

实现这一点的一种方法是首先将字符串 ArrayList 连接到一个分隔字符串中,但我强烈建议不要这样做。 通常,将列表放入 table 的单个字段中是不好的做法。您应该做的是为症状创建一个单独的 table,与评估具有一对多关系。

在使用 JPA 设计对象时,您需要注意非规范化。对于您的情况,请考虑以下问题:

  1. 如果您想查询具有特定症状的评估会怎样?
  2. 如果您想查询所有症状的列表会怎样?
  3. 如果您想用一些其他细节(例如症状出现的日期)来扩展症状,会发生什么情况?

如果您曾经尝试将某些内容的集合添加到数据库字段中,那么 99.9999% 的情况都是您做错了。症状应该是它们自己的实体,并且您在评估和症状之间具有一对多或多对多关系,具体取决于您的需要。

编辑:

为了进一步阐明我的答案,在设计对象时类考虑一个字段是一个值对象还是一个实体。值对象是不能进一步分解的东西,可以用基本类型表示,例如 Date、String、Int 等。一些示例可以是对象 ID、名称、phone 数字等。

实体是可以进一步扩展的对象,例如您创建的 Evaluations 对象。在评估中,您有一个症状列表,并且您将症状视为一个值对象。它是一个值对象吗?我可以立即想到一些您可以放入 Symptom 对象的附加字段,并且通过按照您的方式对症状进行非规范化,您还将大量重复数据输入到数据库中。

在你的实现中包含["Fever", "Cough"]的评估对象将作为一个字段输入到数据库中。但是另一个包含相同症状的评估对象将被输入到该评估的数据库中,因为您没有外键依赖项或单独的 table 表示症状。加上无法查询与评估相关的症状,或者无法自行查询症状。