gpt4 book ai didi

java - 如何在 Jackson 中以增量方式从文件中读取 json 序列?

转载 作者:行者123 更新时间:2023-11-29 05:13:12 26 4
gpt4 key购买 nike

一个过程正在创建一个日志,它是一系列 json 文档。另一个进程想要在日志生成时读取它(类似于 tail -f)并使用 Jackson JsonParser(和 ObjectReader)解析它。 Reader.read 暂时返回-1 时,如何让 JsonParser 或 MappingIterator 不自行关闭?

尝试在 EOF 经常出现的 Reader 上创建 JsonReader.readValues 的示例:

import static org.junit.Assert.*;

import java.io.IOException;
import java.io.Reader;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Objects;

import org.junit.Test;

import com.fasterxml.jackson.annotation.JsonCreator;
import com.fasterxml.jackson.annotation.JsonProperty;
import com.fasterxml.jackson.databind.MappingIterator;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.ObjectReader;

public class ParitalParserTest {
/** Reader that temporarily returns -1 (EOF) to simulate growing file */
private static class StringReaderWithEofs extends Reader {
public final String string;
private int pos;
private int limit;
public StringReaderWithEofs(String string) {
this.string = string;
}
public void limit(int limit) {
this.limit = limit;
}
@Override public int read(char[] cbuf, int off, int len) throws IOException {
int newPos = Math.min(pos + len, limit);
string.getChars(pos, newPos, cbuf, off);
int r = newPos - this.pos;
if (r == 0) r = -1;
this.pos = newPos;
return r;
}
@Override public void close() throws IOException {}
}
/** POJO log event */
public static class LogEvent {
private final String message;
@JsonCreator
public LogEvent(@JsonProperty("message") String message) {
this.message = message;
}
public String getMessage() {
return message;
}
@Override public int hashCode() {
return Objects.hash(message);
}
@Override public boolean equals(Object obj) {
if (! (obj instanceof LogEvent) || obj.getClass() != this.getClass()) return false;
LogEvent o = (LogEvent) obj;
return Objects.equals(message, o.message);
}
@Override public String toString() {
return String.format("%s{message=%s}", getClass().getSimpleName(), message);
}

}
public void test(boolean withEofs) throws IOException {
ObjectMapper mapper = new ObjectMapper();
ObjectReader jsonReader = mapper.reader(LogEvent.class);
String one = "{\"message\":\"one\"}\n";
String two = "{\"message\":\"two\"}\n";
String three = "{\"message\":\"three\"}\n";
StringReaderWithEofs reader = new StringReaderWithEofs(one + two + three);
MappingIterator<LogEvent> it = null;
List<LogEvent> values = new ArrayList<>();
int firstLimit = withEofs ? 1 : reader.string.length();
int lastLimit = reader.string.length();
for (int i = firstLimit; i <= lastLimit; i++) {
reader.limit(i);
if (it == null) {
it = jsonReader.readValues(reader);
}
while (it.hasNext()) {
LogEvent value = it.next();
values.add(value);
}
}
List<LogEvent> expected = Arrays.asList(new LogEvent("one"), new LogEvent("two"), new LogEvent("three"));
assertEquals(expected, values);
}
@Test public void testReadFullFile() throws IOException {
test(false);
}
@Test public void testReadPartialFile() throws IOException {
test(true);
}

}

最佳答案

这有点棘手,但我可能会尝试创建一个 Reader,如果没有可用数据,它实际上会阻塞、 hibernate 、检查等等。这样 Jackson 就不必担心内容不足。

ReaderInputStream 返回 0 实际上没有帮助,因为解析器无能为力,因为实际上没有办法正确阻止。或者,换句话说:底层读取器或流必须尝试读取至少一个字节(如果有一个或多个请求),如果没有找到则阻塞;如果没有可用的,则返回 -1。

关于java - 如何在 Jackson 中以增量方式从文件中读取 json 序列?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27536551/

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