gpt4 book ai didi

java - 如何正确序列化和反序列化 CSV?

转载 作者:行者123 更新时间:2023-11-29 06:54:41 27 4
gpt4 key购买 nike

我一直在尝试将一个对象序列化为 CSV String,但该对象包含一个 List 并且 @JsonUnwrapped 不起作用在 List 对象上。

预期样本输出:

color,part.name\n
red,gearbox\n
red,door\n
red,bumper

实际输出:

com.fasterxml.jackson.core.JsonGenerationException: Unrecognized column 'name':

这是我的代码:(大部分是 2 个 POJO 的)

import com.fasterxml.jackson.annotation.JsonAutoDetect;
import com.fasterxml.jackson.annotation.JsonFormat;
import com.fasterxml.jackson.annotation.JsonInclude;
import com.fasterxml.jackson.annotation.JsonProperty;
import com.fasterxml.jackson.annotation.JsonPropertyOrder;
import com.fasterxml.jackson.annotation.JsonRootName;
import com.fasterxml.jackson.dataformat.csv.CsvMapper;
import com.fasterxml.jackson.dataformat.csv.CsvSchema;
import com.fasterxml.jackson.dataformat.xml.annotation.JacksonXmlElementWrapper;
import java.io.IOException;
import static java.util.Arrays.asList;
import java.util.List;
import java.util.logging.Level;
import java.util.logging.Logger;

public class NestedWrapping {

@JsonRootName("Car")
@JsonInclude(JsonInclude.Include.NON_DEFAULT)
@JsonAutoDetect(fieldVisibility = JsonAutoDetect.Visibility.ANY, getterVisibility = JsonAutoDetect.Visibility.NONE, setterVisibility = JsonAutoDetect.Visibility.NONE)
@JsonPropertyOrder({"color"})
public static class Car {

@JsonProperty("color")
private String color;

@JsonFormat(shape = JsonFormat.Shape.STRING)
@JacksonXmlElementWrapper(useWrapping = false)
private List<Part> parts;

public String getColor() {
return color;
}

public void setColor(String color) {
this.color = color;
}

public List<Part> getParts() {
return parts;
}

public void setParts(List<Part> parts) {
this.parts = parts;
}

}

@JsonInclude(JsonInclude.Include.NON_DEFAULT)
@JsonAutoDetect(fieldVisibility = JsonAutoDetect.Visibility.ANY, getterVisibility = JsonAutoDetect.Visibility.NONE, setterVisibility = JsonAutoDetect.Visibility.NONE)
@JsonPropertyOrder({
"name"
})
public static class Part {

@JsonProperty("name")
private String name;

public String getName() {
return name;
}

public void setName(String name) {
this.name = name;
}

}

public static void main(String args[]) {
try {
Car car = new Car();
car.setColor("red");
Part part1 = new Part();
part1.setName("geabox");
Part part2 = new Part();
part2.setName("door");
Part part3 = new Part();
part3.setName("bumper");
car.setParts(asList(part1, part2, part3));
System.out.println("serialized: " + serialize(car, Car.class, true));
} catch (IOException ex) {
Logger.getLogger(NestedWrapping.class.getName()).log(Level.SEVERE, null, ex);
}
}

public static final synchronized String serialize(final Object object, final Class type, final Boolean withHeaders) throws IOException {
CsvMapper csvMapper = new CsvMapper();
CsvSchema csvSchema;
if (withHeaders) {
csvSchema = csvMapper.schemaFor(type).withHeader();
} else {
csvSchema = csvMapper.schemaFor(type).withoutHeader();
}
return csvMapper.writer(csvSchema).writeValueAsString(object);
}

}

我尝试的一切似乎都不起作用,我已经阅读了所有 post关于该主题的 stackoverflow 和 github,但我找不到有效的解决方案。

对于我无缘无故留下的任何无意义的注释,我深表歉意,如果您用代码回答,请随时删除它们。

最佳答案

根据错误,我想相信它与您的 Car 架构有关,其中包含 {"color"} 的列取自@JsonPropertyOrderCar而不是 "name"值(value)。

您可能想添加 "parts"在那里,但你会得到与 "name" 相同的错误不是该架构的一部分。

对您的代码进行一些更改后,我能够序列化和反序列化 Car目的。

部分

在这里,经过一些其他更改后,它需要一个具有单个 String 值的构造函数,因此添加

@JsonPropertyOrder({"name"})
public static class Part {
@JsonProperty("name")
private String name;

public Part() {
this("");
}

public Part(String partJSON) {
// TODO: Unserialize the parameter... it is a serialized Part string...
this.name = partJSON;
}

汽车

在这里,您需要实现一个方法来转换 List<Part>手动转换为 CSV 可读格式。

这样的方法看起来像这样

@JsonGetter("parts")
public String getPartString() {
String separator = ";";
StringBuilder sb = new StringBuilder();

Iterator<Part> iter = this.parts.iterator();
while (iter.hasNext()) {
Part p = iter.next();
sb.append(p.getName());

if (iter.hasNext())
sb.append(separator);
}

return sb.toString();
}

此外,不要忘记在类的顶部修复模式

@JsonPropertyOrder({"color", "parts"})
public static class Car {

@JsonProperty("color")
private String color;
@JsonProperty("parts")
private List<Part> parts;

public Car() {
this.parts = new ArrayList<>();
}

序列化

您可以更改您的 serialize方法将类的类型作为泛型类型参数而不是显式 Class像这样。

public static final synchronized <T> String serialize(final T object, final Boolean withHeaders) throws IOException {
CsvMapper csvMapper = new CsvMapper();
CsvSchema csvSchema = csvMapper.schemaFor(object.getClass());

if (withHeaders) {
csvSchema = csvSchema.withHeader();
} else {
csvSchema = csvSchema.withoutHeader();
}

return csvMapper.writer(csvSchema).writeValueAsString(object);
}

主编

现在,如果你序列化一个 Car ,你应该看到

color,parts
red,gearbox;door;bumper

主要读者

然后读取 CSV 字符串并遍历 Car.getParts()

Car car = mapper.readerFor(Car.class).with(csvSchema).readValue(csv);

for (Part p : car.getParts()) {
System.out.println(p.getName());
}
gearbox
door
bumper

关于java - 如何正确序列化和反序列化 CSV?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36943511/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com