gpt4 book ai didi

java - jackson-dataformat-csv:没有 POJO 的映射数值

转载 作者:塔克拉玛干 更新时间:2023-11-02 19:06:03 24 4
gpt4 key购买 nike

我正在尝试使用 jackson-dataformat-csv 解析 CSV 文件,我想将数字列映射到 Number java 类型。

CsvSchema schema = CsvSchema.builder().setUseHeader(true)
.addColumn("firstName", CsvSchema.ColumnType.STRING)
.addColumn("lastName", CsvSchema.ColumnType.STRING)
.addColumn("age", CsvSchema.ColumnType.NUMBER)
.build();

CsvMapper csvMapper = new CsvMapper();

MappingIterator<Map<String, Object>> mappingIterator = csvMapper
.readerFor(Map.class)
.with(schema)
.readValues(is);

while (mappingIterator.hasNext()) {
Map<String, Object> entryMap = mappingIterator.next();
Number age = (Number) entryMap.get("age");
}

我期望 entryMap.get("age") 应该是一个 Number,但我得到的是 String

我的 CSV 文件:

firstName,lastName,age
John,Doe,21
Error,Name,-10

我知道 CsvSchema 适用于 POJO,但我需要处理任意 CSV 模式,因此我无法为每种情况创建一个新的 java 类。

有什么方法可以将 CSV 解析为类型化的 MapArray

最佳答案

目前无法使用 CsvSchema 配置 Map 反序列化。进程使用 com.fasterxml.jackson.databind.deser.std.MapDeserializer 现在不检查架构。我们可以编写自定义 Map 反序列化器。 GitHub上有个问题:CsvMapper does not respect CsvSchema.ColumnType when using @JsonAnySetter cowtowncoder 回答的地方:

At this point schema type is not used much for anything, but I agree it should.

编辑

我决定仔细看看我们可以用 com.fasterxml.jackson.databind.deser.std.MapDeserializer 在幕后使用这一事实做些什么。实现自定义的 Map 反序列化器来处理类型将很难实现和注册,但我们可以使用有关 ValueInstantiator 的知识。让我们定义新的 Map 类型,它知道如何处理 ColumnType 信息:

class CsvMap extends HashMap<String, Object> {

private final CsvSchema schema;
private final NumberFormat numberFormat = NumberFormat.getInstance();

public CsvMap(CsvSchema schema) {
this.schema = schema;
}

@Override
public Object put(String key, Object value) {
value = convertIfNeeded(key, value);
return super.put(key, value);
}

private Object convertIfNeeded(String key, Object value) {
CsvSchema.Column column = schema.column(key);
if (column.getType() == CsvSchema.ColumnType.NUMBER) {
try {
return numberFormat.parse(value.toString());
} catch (ParseException e) {
// leave it as it is
}
}

return value;
}
}

对于没有no-arg构造函数的新类型,我们应该创建新的ValueInstantiator:

class CsvMapInstantiator extends ValueInstantiator.Base {

private final CsvSchema schema;

public CsvMapInstantiator(CsvSchema schema) {
super(CsvMap.class);
this.schema = schema;
}

@Override
public Object createUsingDefault(DeserializationContext ctxt) {
return new CsvMap(schema);
}

@Override
public boolean canCreateUsingDefault() {
return true;
}
}

示例用法:

import com.fasterxml.jackson.databind.DeserializationContext;
import com.fasterxml.jackson.databind.MappingIterator;
import com.fasterxml.jackson.databind.ObjectReader;
import com.fasterxml.jackson.databind.deser.ValueInstantiator;
import com.fasterxml.jackson.databind.module.SimpleModule;
import com.fasterxml.jackson.dataformat.csv.CsvMapper;
import com.fasterxml.jackson.dataformat.csv.CsvSchema;

import java.io.File;
import java.io.IOException;
import java.text.NumberFormat;
import java.text.ParseException;
import java.util.HashMap;

public class CsvApp {

public static void main(String[] args) throws IOException {
File csvFile = new File("./resource/test.csv").getAbsoluteFile();

CsvSchema schema = CsvSchema.builder()
.addColumn("firstName", CsvSchema.ColumnType.STRING)
.addColumn("lastName", CsvSchema.ColumnType.STRING)
.addColumn("age", CsvSchema.ColumnType.NUMBER)
.build().withHeader();

// Create schema aware map module
SimpleModule csvMapModule = new SimpleModule();
csvMapModule.addValueInstantiator(CsvMap.class, new CsvMapInstantiator(schema));

// register map
CsvMapper csvMapper = new CsvMapper();
csvMapper.registerModule(csvMapModule);

// get reader for CsvMap + schema
ObjectReader objectReaderWithSchema = csvMapper
.readerWithSchemaFor(CsvMap.class)
.with(schema);

MappingIterator<CsvMap> mappingIterator = objectReaderWithSchema.readValues(csvFile);

while (mappingIterator.hasNext()) {
CsvMap entryMap = mappingIterator.next();

Number age = (Number) entryMap.get("age");
System.out.println(age + " (" + age.getClass() + ")");
}
}
}

下面CSV 负载的上面代码:

firstName,lastName,age
John,Doe,21
Error,Name,-10.1

打印:

21 (class java.lang.Long)
-10.1 (class java.lang.Double)

这看起来像是 hack,但我想展示这种可能性。

关于java - jackson-dataformat-csv:没有 POJO 的映射数值,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55110124/

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