SPR-16351
родолжение моего ответаВот, Хотя у моей бизнес-логики нет проблем, оказалось, что я не использую NettyByteBuf
, Как только я обновил свой тестовый код для использованияByteBuf
Я столкнулся с бесконечной петлейIllegalReferenceCountException, Я признаю, что был новичком в Netty, но это не оправдывает возвращение во времена ручного распределения и выпуска ресурсов. GC был создан, чтобы избежать этого беспорядка. Дискотека, кто-нибудь? Как насчет Белл Боттомс тогда?
public class StringDecoder extends AbstractDecoder<String> {
private static final IntPredicate NEWLINE_DELIMITER = b -> b == '\n' || b == '\r';
@Override
public Flux<String> decode(Publisher<DataBuffer> publisher, ResolvableType elementType, MimeType mimeType, Map<String, Object> hints) {
return Flux.from(publisher)
.scan(Tuples.<Flux<DataBuffer>, Optional<DataBuffer>>of(Flux.empty(), Optional.empty()),
(acc, buffer) -> {
List<DataBuffer> results = new ArrayList<>();
int startIdx = 0, endIdx = 0, limit = buffer.readableByteCount();
Optional<DataBuffer> incomplete = acc.getT2();
while (startIdx < limit && endIdx != -1) {
endIdx = buffer.indexOf(NEWLINE_DELIMITER, startIdx);
int length = (endIdx == -1 ? limit : endIdx) - startIdx;
DataBuffer slice = buffer.slice(startIdx, length);
DataBuffer tmp = incomplete.map(b -> b.write(slice))
.orElse(buffer.factory().allocateBuffer(length).write(slice));
tmp = DataBufferUtils.retain(tmp);
if (endIdx != -1) {
startIdx = endIdx + 1;
results.add(tmp);
incomplete = Optional.empty();
} else {
incomplete = Optional.of(tmp);
}
}
releaseBuffer(buffer);
return Tuples.of(Flux.fromIterable(results), incomplete);
})
.flatMap(t -> {
t.getT2().ifPresent(this::releaseBuffer);
return t.getT1();
})
.map(buffer -> {
// charset resolution should in general use supplied mimeType
String s = UTF_8.decode(buffer.asByteBuffer()).toString();
releaseBuffer(buffer);
return s;
})
.log();
}
private void releaseBuffer(DataBuffer buffer) {
boolean release = DataBufferUtils.release(buffer);
if (release) {
System.out.println("Buffer was released.");
}
}
}
public class StringDecoderTest {
private StringDecoder stringDecoder = new StringDecoder();
DataBufferFactory dataBufferFactory = new NettyDataBufferFactory(UnpooledByteBufAllocator.DEFAULT);
@Test
public void testDecode() {
Flux<DataBuffer> pub = Flux.just("abc\n", "abc", "def\n", "abc", "def\nxyz\n", "abc", "def", "xyz\n")
.map(s -> dataBufferFactory.wrap(s.getBytes(UTF_8)));
StepVerifier.create(stringDecoder.decode(pub, null, null, null))
.expectNext("abc", "abcdef", "abcdef", "xyz", "abcdefxyz")
.verifyComplete();
}
}
Я продолжаю получать:
[ERROR] (main) onError(io.netty.util.IllegalReferenceCountException: refCnt: 0)
[ERROR] (main) - io.netty.util.IllegalReferenceCountException: refCnt: 0
io.netty.util.IllegalReferenceCountException: refCnt: 0
at io.netty.buffer.AbstractByteBuf.ensureAccessible(AbstractByteBuf.java:1415)
at io.netty.buffer.UnpooledHeapByteBuf.nioBuffer(UnpooledHeapByteBuf.java:314)
at io.netty.buffer.AbstractUnpooledSlicedByteBuf.nioBuffer(AbstractUnpooledSlicedByteBuf.java:434)
at io.netty.buffer.CompositeByteBuf.nioBuffers(CompositeByteBuf.java:1496)
at io.netty.buffer.CompositeByteBuf.nioBuffer(CompositeByteBuf.java:1468)
at io.netty.buffer.AbstractByteBuf.nioBuffer(AbstractByteBuf.java:1205)
at org.springframework.core.io.buffer.NettyDataBuffer.asByteBuffer(NettyDataBuffer.java:234)
at org.abhijitsarkar.java.StringDecoder.lambda$decode$4(StringDecoder.java:61)