ref[IEntityWrapper]: wrap entity method

This commit is contained in:
godotg
2024-07-01 15:12:35 +08:00
parent 688c2a9917
commit e13cdd424a
7 changed files with 186 additions and 162 deletions
+26 -33
View File
@@ -18,21 +18,20 @@ import com.mongodb.client.model.Filters;
import com.mongodb.client.model.ReplaceOneModel;
import com.zfoo.event.manager.EventBus;
import com.zfoo.orm.OrmContext;
import com.zfoo.orm.anno.Version;
import com.zfoo.orm.cache.persister.IOrmPersister;
import com.zfoo.orm.cache.persister.PNode;
import com.zfoo.orm.cache.version.VersionDefault;
import com.zfoo.orm.cache.version.VersionReflect;
import com.zfoo.orm.cache.version.EnhanceUtils;
import com.zfoo.orm.cache.version.IVersion;
import com.zfoo.orm.cache.wrapper.EnhanceUtils;
import com.zfoo.orm.cache.wrapper.EntityWrapper;
import com.zfoo.orm.cache.wrapper.IEntityWrapper;
import com.zfoo.orm.model.EntityDef;
import com.zfoo.orm.model.IEntity;
import com.zfoo.orm.query.Page;
import com.zfoo.protocol.collection.ArrayUtils;
import com.zfoo.protocol.collection.CollectionUtils;
import com.zfoo.protocol.exception.RunException;
import com.zfoo.protocol.model.Pair;
import com.zfoo.protocol.util.*;
import com.zfoo.protocol.util.AssertionUtils;
import com.zfoo.protocol.util.GraalVmUtils;
import com.zfoo.protocol.util.ThreadUtils;
import com.zfoo.scheduler.manager.SchedulerBus;
import com.zfoo.scheduler.util.LazyCache;
import com.zfoo.scheduler.util.TimeUtils;
@@ -59,29 +58,23 @@ public class EntityCache<PK extends Comparable<PK>, E extends IEntity<PK>> imple
private final LazyCache<PK, PNode<PK, E>> cache;
private IVersion version = VersionDefault.DEFAULT;
private IEntityWrapper wrapper;
public EntityCache(EntityDef entityDef) {
var clazz = entityDef.getClazz();
// 创建CacheVersion
var versionFields = ReflectionUtils.getFieldsByAnnoInPOJOClass(clazz, Version.class);
if (ArrayUtils.isNotEmpty(versionFields)) {
var filed = versionFields[0];
var getMethod = ReflectionUtils.getMethodByNameInPOJOClass(clazz, FieldUtils.fieldToGetMethod(clazz, filed));
var setMethod = ReflectionUtils.getMethodByNameInPOJOClass(clazz, FieldUtils.fieldToSetMethod(clazz, filed), filed.getType());
var cacheVersionReflect = new VersionReflect(clazz, getMethod, setMethod, filed.getName());
if (GraalVmUtils.isGraalVM()) {
version = cacheVersionReflect;
} else {
try {
version = EnhanceUtils.createVersion(cacheVersionReflect);
} catch (Exception e) {
throw new RuntimeException(e);
}
var entityWrapper = new EntityWrapper(entityDef.getClazz());
if (GraalVmUtils.isGraalVM()) {
wrapper = entityWrapper;
} else {
try {
wrapper = EnhanceUtils.createEntityWrapper(entityWrapper);
} catch (Exception e) {
throw new RuntimeException(e);
}
}
var removeCallback = new BiConsumer<Pair<PK, PNode<PK, E>>, LazyCache.RemovalCause>() {
@Override
public void accept(Pair<PK, PNode<PK, E>> pair, LazyCache.RemovalCause removalCause) {
@@ -104,11 +97,11 @@ public class EntityCache<PK extends Comparable<PK>, E extends IEntity<PK>> imple
public void run() {
var collection = OrmContext.getOrmManager().getCollection(entityClass);
var currentVersion = version.gvs(entity);
version.svs(entity, currentVersion + 1);
var version = wrapper.gvs(entity);
wrapper.svs(entity, version + 1);
var filter = version.gvs(entity) > 0
? Filters.and(Filters.eq("_id", entity.id()), Filters.eq(version.name(), currentVersion))
var filter = wrapper.gvs(entity) > 0
? Filters.and(Filters.eq("_id", entity.id()), Filters.eq(wrapper.versionFieldName(), version))
: Filters.eq("_id", entity.id());
var result = collection.replaceOne(filter, entity);
if (result.getModifiedCount() <= 0) {
@@ -342,11 +335,11 @@ public class EntityCache<PK extends Comparable<PK>, E extends IEntity<PK>> imple
var batchList = currentUpdateList.stream()
.map(it -> {
var currentVersion = version.gvs(it);
version.svs(it, currentVersion + 1);
var version = wrapper.gvs(it);
wrapper.svs(it, version + 1);
var filter = version.gvs(it) > 0
? Filters.and(Filters.eq("_id", it.id()), Filters.eq(version.name(), currentVersion))
var filter = wrapper.gvs(it) > 0
? Filters.and(Filters.eq("_id", it.id()), Filters.eq(wrapper.versionFieldName(), version))
: Filters.eq("_id", it.id());
return new ReplaceOneModel<>(filter, it);
@@ -393,8 +386,8 @@ public class EntityCache<PK extends Comparable<PK>, E extends IEntity<PK>> imple
}
// 如果没有版本号,则直接更新数据库
var entityVersion = version.gvs(entity);
var dbEntityVersion = version.gvs(dbEntity);
var entityVersion = wrapper.gvs(entity);
var dbEntityVersion = wrapper.gvs(dbEntity);
if (entityVersion <= 0) {
OrmContext.getAccessor().update(entity);
continue;
@@ -1,39 +0,0 @@
/*
* Copyright (C) 2020 The zfoo Authors
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except
* in compliance with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software distributed under the License is distributed
* on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and limitations under the License.
*/
package com.zfoo.orm.cache.version;
import com.zfoo.orm.model.IEntity;
/**
* @author godotg
*/
public class VersionDefault implements IVersion {
public static final VersionDefault DEFAULT = new VersionDefault();
@Override
public String name() {
return "version";
}
@Override
public long gvs(IEntity<?> entity) {
return 0;
}
@Override
public void svs(IEntity<?> entity, long vs) {
}
}
@@ -1,68 +0,0 @@
/*
* Copyright (C) 2020 The zfoo Authors
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except
* in compliance with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software distributed under the License is distributed
* on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and limitations under the License.
*/
package com.zfoo.orm.cache.version;
import com.zfoo.orm.model.IEntity;
import com.zfoo.protocol.util.ReflectionUtils;
import java.lang.reflect.Method;
/**
* @author godotg
*/
public class VersionReflect implements IVersion {
private Class<? extends IEntity<?>> entityClass;
private Method getVersionMethod;
private Method setVersionMethod;
private String versionFiled;
public VersionReflect(Class<? extends IEntity<?>> entityClass, Method getVersionMethod, Method setVersionMethod, String versionFiled) {
this.entityClass = entityClass;
this.getVersionMethod = getVersionMethod;
this.setVersionMethod = setVersionMethod;
this.versionFiled = versionFiled;
}
@Override
public String name() {
return versionFiled;
}
@Override
public long gvs(IEntity<?> entity) {
return (long) ReflectionUtils.invokeMethod(entity, getVersionMethod);
}
@Override
public void svs(IEntity<?> entity, long vs) {
ReflectionUtils.invokeMethod(entity, setVersionMethod, vs);
}
public Class<? extends IEntity<?>> getEntityClass() {
return entityClass;
}
public Method getGetVersionMethod() {
return getVersionMethod;
}
public Method getSetVersionMethod() {
return setVersionMethod;
}
public String getVersionFiled() {
return versionFiled;
}
}
@@ -10,14 +10,16 @@
* See the License for the specific language governing permissions and limitations under the License.
*/
package com.zfoo.orm.cache.version;
package com.zfoo.orm.cache.wrapper;
import com.zfoo.orm.model.IEntity;
import com.zfoo.orm.schema.NamespaceHandler;
import com.zfoo.protocol.util.StringUtils;
import com.zfoo.protocol.util.UuidUtils;
import javassist.*;
import org.bson.types.ObjectId;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;
import java.lang.reflect.Method;
import java.lang.reflect.Modifier;
@@ -30,7 +32,7 @@ public abstract class EnhanceUtils {
static {
// 适配Tomcat因为Tomcat不是用的默认的类加载器而Javassist用的是默认的加载器
var classArray = new Class<?>[]{
IVersion.class
IEntityWrapper.class
};
var classPool = ClassPool.getDefault();
@@ -43,42 +45,68 @@ public abstract class EnhanceUtils {
}
}
public static IVersion createVersion(VersionReflect cacheVersion) throws NotFoundException, CannotCompileException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
public static String rawObjectId(Field idField) {
var idType = idField.getType();
if (idType == int.class || idType == Integer.class) {
return "((Integer)$1).intValue()";
} else if (idType == long.class || idType == Long.class) {
return "((Long)$1).longValue()";
} else if (idType == float.class || idType == Float.class) {
return "((Float)$1).floatValue()";
} else if (idType == double.class || idType == Double.class) {
return "((Double)$1).floatValue()";
} else if (idType == String.class) {
return "(String)$1";
} else {
return StringUtils.format("({})$1", ObjectId.class.getName());
}
}
public static IEntityWrapper createEntityWrapper(EntityWrapper entityWrapper) throws NotFoundException, CannotCompileException, NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
var classPool = ClassPool.getDefault();
Class<?> clazz = cacheVersion.getEntityClass();
Method getVersionMethod = cacheVersion.getGetVersionMethod();
Method setVersionMethod = cacheVersion.getSetVersionMethod();
Class<?> clazz = entityWrapper.getEntityClass();
Field idField = entityWrapper.getIdField();
Method setIdMethod = entityWrapper.getSetIdMethod();
Method getVersionMethod = entityWrapper.getGetVersionMethod();
Method setVersionMethod = entityWrapper.getSetVersionMethod();
// 定义类名称
CtClass enhanceClazz = classPool.makeClass(EnhanceUtils.class.getName() + StringUtils.capitalize(NamespaceHandler.ORM) + UuidUtils.getLocalIntId());
enhanceClazz.addInterface(classPool.get(IVersion.class.getName()));
enhanceClazz.addInterface(classPool.get(IEntityWrapper.class.getName()));
// 定义类实现的接口方法name
CtMethod nameMethod = new CtMethod(classPool.get(String.class.getName()), "name", null, enhanceClazz);
CtMethod newEntityMethod = new CtMethod(classPool.get(IEntity.class.getName()), "newEntity", classPool.get(new String[]{Object.class.getName()}), enhanceClazz);
newEntityMethod.setModifiers(Modifier.PUBLIC + Modifier.FINAL);
String newEntityMethodBody = StringUtils.format("{ {} entity = new {}(); entity.{}({}); return entity; }", clazz.getName(), clazz.getName(), setIdMethod.getName(), rawObjectId(idField));
newEntityMethod.setBody(newEntityMethodBody);
enhanceClazz.addMethod(newEntityMethod);
// 定义类实现的接口方法name
CtMethod nameMethod = new CtMethod(classPool.get(String.class.getName()), "versionFieldName", null, enhanceClazz);
nameMethod.setModifiers(Modifier.PUBLIC + Modifier.FINAL);
String nameMethodBody = StringUtils.format("{ return \"{}\"; }", cacheVersion.name());
String nameMethodBody = StringUtils.format("{ return \"{}\"; }", entityWrapper.versionFieldName());
nameMethod.setBody(nameMethodBody);
enhanceClazz.addMethod(nameMethod);
// 定义类实现的接口方法gvs
CtMethod gvsMethod = new CtMethod(classPool.get(long.class.getName()), "gvs", classPool.get(new String[]{IEntity.class.getName()}), enhanceClazz);
gvsMethod.setModifiers(Modifier.PUBLIC + Modifier.FINAL);
String gvsMethodBody = StringUtils.format("{ return (({})$1).{}(); }", clazz.getName(), getVersionMethod.getName());
String gvsMethodBody = getVersionMethod == null ? "{ return 0L; }" : StringUtils.format("{ return (({})$1).{}(); }", clazz.getName(), getVersionMethod.getName());
gvsMethod.setBody(gvsMethodBody);
enhanceClazz.addMethod(gvsMethod);
// 定义类实现的接口方法svs
CtMethod svsMethod = new CtMethod(classPool.get(void.class.getName()), "svs", classPool.get(new String[]{IEntity.class.getName(), long.class.getName()}), enhanceClazz);
svsMethod.setModifiers(Modifier.PUBLIC + Modifier.FINAL);
String svsMethodBody = StringUtils.format("{ (({})$1).{}($2); }", clazz.getName(), setVersionMethod.getName());
String svsMethodBody = setVersionMethod == null ? "{}" : StringUtils.format("{ (({})$1).{}($2); }", clazz.getName(), setVersionMethod.getName());
svsMethod.setBody(svsMethodBody);
enhanceClazz.addMethod(svsMethod);
// 释放缓存
enhanceClazz.detach();
Class<?> resultClazz = enhanceClazz.toClass(IVersion.class);
return (IVersion) resultClazz.newInstance();
Class<?> resultClazz = enhanceClazz.toClass(IEntityWrapper.class);
return (IEntityWrapper) resultClazz.newInstance();
}
}
@@ -0,0 +1,106 @@
/*
* Copyright (C) 2020 The zfoo Authors
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except
* in compliance with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software distributed under the License is distributed
* on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and limitations under the License.
*/
package com.zfoo.orm.cache.wrapper;
import com.zfoo.orm.anno.Id;
import com.zfoo.orm.anno.Version;
import com.zfoo.orm.model.IEntity;
import com.zfoo.protocol.collection.ArrayUtils;
import com.zfoo.protocol.util.FieldUtils;
import com.zfoo.protocol.util.ReflectionUtils;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
/**
* @author godotg
*/
public class EntityWrapper implements IEntityWrapper {
private Class<? extends IEntity<?>> entityClass;
private Field idField;
private Field versionField;
private Method setIdMethod;
private Method getVersionMethod;
private Method setVersionMethod;
public EntityWrapper(Class<? extends IEntity<?>> entityClass) {
this.entityClass = entityClass;
var idFields = ReflectionUtils.getFieldsByAnnoInPOJOClass(entityClass, Id.class);
ReflectionUtils.makeAccessible(idFields[0]);
this.idField = idFields[0];
this.setIdMethod = ReflectionUtils.getMethodByNameInPOJOClass(entityClass, FieldUtils.fieldToSetMethod(entityClass, idField), idField.getType());
var versionFields = ReflectionUtils.getFieldsByAnnoInPOJOClass(entityClass, Version.class);
if (ArrayUtils.isNotEmpty(versionFields)) {
this.versionField = versionFields[0];
var getMethod = ReflectionUtils.getMethodByNameInPOJOClass(entityClass, FieldUtils.fieldToGetMethod(entityClass, versionField));
var setMethod = ReflectionUtils.getMethodByNameInPOJOClass(entityClass, FieldUtils.fieldToSetMethod(entityClass, versionField), versionField.getType());
this.getVersionMethod = getMethod;
this.setVersionMethod = setMethod;
}
}
@Override
public IEntity<?> newEntity(Object id) {
var entity = ReflectionUtils.newInstance(entityClass);
ReflectionUtils.invokeMethod(entity, setIdMethod, id);
return entity;
}
@Override
public String versionFieldName() {
return idField.getName();
}
@Override
public long gvs(IEntity<?> entity) {
if (getVersionMethod == null) {
return 0;
}
return (long) ReflectionUtils.invokeMethod(entity, getVersionMethod);
}
@Override
public void svs(IEntity<?> entity, long vs) {
if (setVersionMethod == null) {
return;
}
ReflectionUtils.invokeMethod(entity, setVersionMethod, vs);
}
public Class<? extends IEntity<?>> getEntityClass() {
return entityClass;
}
public Field getIdField() {
return idField;
}
public Field getVersionField() {
return versionField;
}
public Method getSetIdMethod() {
return setIdMethod;
}
public Method getGetVersionMethod() {
return getVersionMethod;
}
public Method getSetVersionMethod() {
return setVersionMethod;
}
}
@@ -10,7 +10,7 @@
* See the License for the specific language governing permissions and limitations under the License.
*/
package com.zfoo.orm.cache.version;
package com.zfoo.orm.cache.wrapper;
import com.zfoo.orm.model.IEntity;
@@ -18,12 +18,14 @@ import com.zfoo.orm.model.IEntity;
/**
* @author godotg
*/
public interface IVersion {
public interface IEntityWrapper {
IEntity<?> newEntity(Object id);
/**
* version field的名称
*/
String name();
String versionFieldName();
/**
* 一个文档的写入到数据库的version版本version的get和set方法
+8 -6
View File
@@ -51,13 +51,15 @@ public class EntityCachesTest {
var userEntityCaches = (IEntityCache<Long, UserEntity>) OrmContext.getOrmManager().getEntityCaches(UserEntity.class);
for (var i = 1; i <= 10; i++) {
var entity = userEntityCaches.load((long) i);
entity.setE("update" + i);
entity.setC(i);
EventBus.asyncExecute(() -> userEntityCaches.update(entity));
}
for (var j = 1; j <= 10; j++) {
var entity = userEntityCaches.load((long) j);
entity.setE("update" + j);
entity.setC(j);
userEntityCaches.update(entity);
}
ThreadUtils.sleep(60 * TimeUtils.MILLIS_PER_SECOND);
ThreadUtils.sleep(60 * TimeUtils.MILLIS_PER_SECOND);
}
userEntityCaches.load(1L);
}