cn.nukkit.utils.collection.FreezableArrayManager Maven / Gradle / Ivy
package cn.nukkit.utils.collection;
import cn.nukkit.Server;
import cn.nukkit.api.PowerNukkitXOnly;
import cn.nukkit.api.Since;
import org.jetbrains.annotations.NotNull;
import java.util.Arrays;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.atomic.AtomicInteger;
/**
* FreezableArrayManager负责管理所有AutoFreezable的ByteArrayWrapper
* 这包括计算温度,冻结和解冻
*/
@PowerNukkitXOnly
@Since("1.19.50-r1")
public class FreezableArrayManager {
protected ConcurrentHashMap> tickArrayMap;
public final boolean enable;
public final int cycleTick;
/**
* 最大工作时间,如果一直压缩超出这个时间就会放弃接下来其他数组的压缩(冻结)
*/
private int maxCompressionTime = 50;
private final AtomicInteger currentArrayId = new AtomicInteger(0);
private int currentTick;
/**
* 默认温度,新创建的数组温度等于此温度
*/
private final int defaultTemperature;
/**
* 冰点,当可冻结数组的温度低于冰点时有可能被冻结
*/
private final int freezingPoint;
/**
* 绝对零度,任何可冻结数组的温度都不应该低于此温度,等于此温度的可冻结数组有可能被深度冻结
*/
private final int absoluteZero;
/**
* 沸点,一个可冻结数组的温度无论如何加热都不能高于此温度
*/
private final int boilingPoint;
/**
* 熔化热,解冻后的数组温度会等于熔化热
*/
private final int meltingHeat;
/**
* 单次数组读写操作升温
*/
private final int singleOperationHeat;
/**
* 一次批量数组读写操作升温
*/
private final int batchOperationHeat;
private static FreezableArrayManager fallbackInstance = null;
public static FreezableArrayManager getInstance() {
try {
var server = Server.getInstance();
if (server != null) {
var tmp = server.getFreezableArrayManager();
if (tmp != null) {
return tmp;
}
}
} catch (Exception e) {
e.printStackTrace();
}
if (fallbackInstance == null) {
fallbackInstance = new FreezableArrayManager(true, 32, 32, 0, -256, 1024, 16, 1, 32);
System.err.println("Cannot get FreezableArrayManager from Server instance, using a fallback instance!");
}
return fallbackInstance;
}
public FreezableArrayManager(boolean enable, int cycleTick, int defaultTemperature, int freezingPoint, int absoluteZero, int boilingPoint, int meltingHeat, int singleOperationHeat, int batchOperationHeat) {
this.enable = enable;
this.cycleTick = cycleTick;
this.defaultTemperature = defaultTemperature;
this.freezingPoint = freezingPoint;
this.absoluteZero = absoluteZero;
this.tickArrayMap = new ConcurrentHashMap<>(cycleTick + 1, 0.999f);
this.boilingPoint = boilingPoint;
this.meltingHeat = meltingHeat;
this.singleOperationHeat = singleOperationHeat;
this.batchOperationHeat = batchOperationHeat;
}
public int getDefaultTemperature() {
return defaultTemperature;
}
public int getAbsoluteZero() {
return absoluteZero;
}
public int getFreezingPoint() {
return freezingPoint;
}
public int getMeltingHeat() {
return meltingHeat;
}
public int getBoilingPoint() {
return boilingPoint;
}
public int getSingleOperationHeat() {
return singleOperationHeat;
}
public int getBatchOperationHeat() {
return batchOperationHeat;
}
public int getMaxCompressionTime() {
return maxCompressionTime;
}
public FreezableArrayManager setMaxCompressionTime(int maxCompressionTime) {
this.maxCompressionTime = maxCompressionTime;
return this;
}
public ByteArrayWrapper createByteArray(int length) {
if (enable) {
var tmp = new FreezableByteArray(length, this);
var set = tickArrayMap.computeIfAbsent(currentArrayId.getAndIncrement() % cycleTick, (ignore) -> new WeakConcurrentSet<>(WeakConcurrentSet.Cleaner.MANUAL));
set.add(tmp);
return tmp;
} else {
return new PureByteArray(length);
}
}
public ByteArrayWrapper wrapByteArray(@NotNull byte[] array) {
if (enable) {
var tmp = new FreezableByteArray(array, this);
var set = tickArrayMap.computeIfAbsent(currentArrayId.getAndIncrement() % cycleTick, (ignore) -> new WeakConcurrentSet<>(WeakConcurrentSet.Cleaner.MANUAL));
set.add(tmp);
return tmp;
} else {
return new PureByteArray(array);
}
}
public ByteArrayWrapper cloneByteArray(@NotNull byte[] array) {
if (enable) {
var tmp = new FreezableByteArray(Arrays.copyOf(array, array.length), this);
var set = tickArrayMap.computeIfAbsent(currentArrayId.getAndIncrement() % cycleTick, (ignore) -> new WeakConcurrentSet<>(WeakConcurrentSet.Cleaner.MANUAL));
set.add(tmp);
return tmp;
} else {
return new PureByteArray(Arrays.copyOf(array, array.length));
}
}
public void tick() {
currentTick++;
if (!enable) return;
var dt = currentTick % cycleTick;
var set = tickArrayMap.get(dt);
// 冻结数组
var start = System.currentTimeMillis();
// 清理死引用
CompletableFuture.runAsync(() -> set.parallelForeach(e -> {
if (e == null) return;
int temp = e.getTemperature();
e.colder(1);
if (temp <= getFreezingPoint() + 1) {
if (System.currentTimeMillis() - start > maxCompressionTime) {
return;
}
if (e.getFreezeStatus() == AutoFreezable.FreezeStatus.NONE || e.getFreezeStatus() == AutoFreezable.FreezeStatus.FREEZE) {
if (e.getTemperature() == absoluteZero) {
e.deepFreeze();
} else {
e.freeze();
}
}
}
}), Server.getInstance().computeThreadPool).thenRun(set::clearDeadReferences);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy