Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
com.jn.agileway.codec.serialization.fst.Fsts Maven / Gradle / Ivy
Go to download
Provide an unified codec API for hession, kryo, protostuff, fst, fes, xson, cbor, jackson, json, etc....
package com.jn.agileway.codec.serialization.fst;
import com.jn.langx.Factory;
import com.jn.langx.annotation.NonNull;
import com.jn.langx.annotation.Nullable;
import com.jn.langx.util.Preconditions;
import com.jn.langx.util.Throwables;
import com.jn.langx.util.collection.Collects;
import com.jn.langx.util.collection.Pipeline;
import com.jn.langx.util.concurrent.threadlocal.ThreadLocalFactory;
import com.jn.langx.util.function.Consumer;
import com.jn.langx.util.function.Consumer2;
import com.jn.langx.util.io.IOs;
import com.jn.langx.util.logging.Loggers;
import org.nustaq.serialization.FSTConfiguration;
import org.nustaq.serialization.FSTObjectInput;
import org.nustaq.serialization.FSTObjectOutput;
import org.slf4j.Logger;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import java.util.Map;
import java.util.ServiceLoader;
import java.util.concurrent.ConcurrentHashMap;
public class Fsts {
private Fsts() {
}
private static final Logger logger = Loggers.getLogger(Fsts.class);
private static final Map fstCustomizerRegistry = new ConcurrentHashMap();
public static final ThreadLocalFactory, FSTConfiguration> fstFactory = new ThreadLocalFactory(new Factory() {
@Override
public FSTConfiguration get(Object o) {
final FSTConfiguration fst = FSTConfiguration.getDefaultConfiguration();
// 用于解决循环依赖
fst.setShareReferences(true);
// 不检查是否实现 Serializable, Externalizable
fst.setStructMode(false);
Collects.forEach(fstCustomizerRegistry, new Consumer2() {
@Override
public void accept(String name, FstCustomizer customizer) {
try {
customizer.customize(fst);
} catch (Throwable ex) {
if (ex instanceof NoClassDefFoundError || ex instanceof ClassNotFoundException) {
logger.error("Error occur when register FST serializers for {}, may be you should append de.javakaffee:kryo-serializers.jar to the classpath", name);
} else {
logger.error("Error occur when register FST customizer for {}", name);
}
}
}
});
return fst;
}
});
public static byte[] serialize(@Nullable T o) throws IOException {
return serialize(fstFactory, o);
}
public static void serialize(@Nullable T o, OutputStream outputStream) throws IOException {
serialize(fstFactory, o, outputStream);
}
public static byte[] serialize(@NonNull Factory, FSTConfiguration> fstFactory, @Nullable T o) throws IOException {
Preconditions.checkNotNull(fstFactory, "the FST factory is null");
return serialize(fstFactory.get(null), o);
}
public static void serialize(@NonNull Factory, FSTConfiguration> fstFactory, @Nullable T o, OutputStream outputStream) throws IOException {
Preconditions.checkNotNull(fstFactory, "the FST factory is null");
serialize(fstFactory.get(null), o, outputStream);
}
public static byte[] serialize(@NonNull FSTConfiguration fst, @Nullable T o) throws IOException {
if (o == null) {
return null;
}
ByteArrayOutputStream bao = null;
try {
bao = new ByteArrayOutputStream();
serialize(fst, o, bao);
return bao.toByteArray();
} finally {
IOs.close(bao);
}
}
public static void serialize(@NonNull FSTConfiguration fst, @NonNull T o, @NonNull OutputStream outputStream) throws IOException {
if (o != null) {
// 底层已经采用ThreadLocal进行了 FSTObjectOutput缓存,用于可重用,所以不必调用 output.close()
FSTObjectOutput output = fst.getObjectOutput(outputStream);
output.writeObject(o, o.getClass());
output.flush();
}
}
public static T deserialize(@Nullable byte[] bytes) {
return deserialize(fstFactory, bytes);
}
public static T deserialize(@Nullable byte[] bytes, @Nullable Class targetClass) {
return deserialize(fstFactory, bytes, targetClass);
}
public static T deserialize(@NonNull Factory, FSTConfiguration> fstFactory, @Nullable byte[] bytes) {
return deserialize(fstFactory, bytes, null);
}
public static T deserialize(@NonNull Factory, FSTConfiguration> fstFactory, @Nullable byte[] bytes, @Nullable Class targetClass) {
Preconditions.checkNotNull(fstFactory, "the FST factory is null");
return deserialize(fstFactory.get(null), bytes, targetClass);
}
public static T deserialize(@NonNull FSTConfiguration fst, @Nullable byte[] bytes) {
return deserialize(fst, bytes, null);
}
public static T deserialize(@NonNull FSTConfiguration fst, @Nullable byte[] bytes, @Nullable Class targetClass) {
if (bytes == null) {
return null;
}
try {
FSTObjectInput input = fst.getObjectInput(bytes);
if (targetClass != null) {
fst.registerClass(targetClass);
return (T) input.readObject(targetClass);
} else {
return (T) input.readObject();
}
} catch (Exception ex) {
throw Throwables.wrapAsRuntimeException(ex);
}
}
static {
ServiceLoader loader = ServiceLoader.load(FstCustomizer.class);
Pipeline.of(loader).forEach(new Consumer() {
@Override
public void accept(FstCustomizer fstCustomizer) {
fstCustomizerRegistry.put(fstCustomizer.getName(), fstCustomizer);
logger.info("Load the FST serializers for {}", fstCustomizer.getName());
}
});
}
public static void registerCustomizer(String name, FstCustomizer customizer) {
fstCustomizerRegistry.put(name, customizer);
}
public static FstCustomizer getCustomizer(String name) {
return fstCustomizerRegistry.get(name);
}
}