mirror of
				https://gitee.com/dromara/hutool.git
				synced 2025-10-26 10:49:27 +08:00 
			
		
		
		
	!1097 修复合成注解在并发环境无法保证正确缓存属性值的问题(#I8CLBJ)
Merge pull request !1097 from Createsequence/fix/I8CLBJ
This commit is contained in:
		| @@ -4,13 +4,13 @@ import cn.hutool.core.annotation.scanner.AnnotationScanner; | |||||||
| import cn.hutool.core.collection.CollUtil; | import cn.hutool.core.collection.CollUtil; | ||||||
| import cn.hutool.core.lang.Assert; | import cn.hutool.core.lang.Assert; | ||||||
| import cn.hutool.core.map.MapUtil; | import cn.hutool.core.map.MapUtil; | ||||||
| import cn.hutool.core.util.ObjectUtil; |  | ||||||
|  |  | ||||||
| import java.lang.annotation.Annotation; | import java.lang.annotation.Annotation; | ||||||
| import java.util.Collection; | import java.util.Collection; | ||||||
| import java.util.Comparator; | import java.util.Comparator; | ||||||
| import java.util.LinkedHashMap; | import java.util.LinkedHashMap; | ||||||
| import java.util.Map; | import java.util.Map; | ||||||
|  | import java.util.Objects; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * {@link AnnotationSynthesizer}的基本实现 |  * {@link AnnotationSynthesizer}的基本实现 | ||||||
| @@ -158,11 +158,19 @@ public abstract class AbstractAnnotationSynthesizer<T> implements AnnotationSynt | |||||||
| 	@SuppressWarnings("unchecked") | 	@SuppressWarnings("unchecked") | ||||||
| 	@Override | 	@Override | ||||||
| 	public <A extends Annotation> A synthesize(Class<A> annotationType) { | 	public <A extends Annotation> A synthesize(Class<A> annotationType) { | ||||||
| 		return (A)synthesizedProxyAnnotations.computeIfAbsent(annotationType, type -> { | 		A annotation = (A)synthesizedProxyAnnotations.get(annotationType); | ||||||
| 			final SynthesizedAnnotation synthesizedAnnotation = synthesizedAnnotationMap.get(annotationType); | 		if (Objects.nonNull(annotation)) { | ||||||
| 			return ObjectUtil.isNull(synthesizedAnnotation) ? | 			return annotation; | ||||||
| 				null : synthesize(annotationType, synthesizedAnnotation); | 		} | ||||||
| 		}); | 		synchronized (synthesizedProxyAnnotations) { | ||||||
|  | 			annotation = (A)synthesizedProxyAnnotations.get(annotationType); | ||||||
|  | 			if (Objects.isNull(annotation)) { | ||||||
|  | 				final SynthesizedAnnotation synthesizedAnnotation = synthesizedAnnotationMap.get(annotationType); | ||||||
|  | 				annotation = synthesize(annotationType, synthesizedAnnotation); | ||||||
|  | 				synthesizedProxyAnnotations.put(annotationType, annotation); | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 		return annotation; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -14,7 +14,7 @@ import java.lang.reflect.Method; | |||||||
|  */ |  */ | ||||||
| public class CacheableAnnotationAttribute implements AnnotationAttribute { | public class CacheableAnnotationAttribute implements AnnotationAttribute { | ||||||
|  |  | ||||||
| 	private boolean valueInvoked; | 	private volatile boolean valueInvoked; | ||||||
| 	private Object value; | 	private Object value; | ||||||
|  |  | ||||||
| 	private boolean defaultValueInvoked; | 	private boolean defaultValueInvoked; | ||||||
| @@ -45,8 +45,12 @@ public class CacheableAnnotationAttribute implements AnnotationAttribute { | |||||||
| 	@Override | 	@Override | ||||||
| 	public Object getValue() { | 	public Object getValue() { | ||||||
| 		if (!valueInvoked) { | 		if (!valueInvoked) { | ||||||
| 			valueInvoked = true; | 			synchronized (this) { | ||||||
| 			value = ReflectUtil.invoke(annotation, attribute); | 				if (!valueInvoked) { | ||||||
|  | 					valueInvoked = true; | ||||||
|  | 					value = ReflectUtil.invoke(annotation, attribute); | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
| 		return value; | 		return value; | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -3,10 +3,10 @@ package cn.hutool.core.annotation; | |||||||
| import cn.hutool.core.lang.Assert; | import cn.hutool.core.lang.Assert; | ||||||
| import cn.hutool.core.map.multi.RowKeyTable; | import cn.hutool.core.map.multi.RowKeyTable; | ||||||
| import cn.hutool.core.map.multi.Table; | import cn.hutool.core.map.multi.Table; | ||||||
| import cn.hutool.core.util.ObjectUtil; |  | ||||||
|  |  | ||||||
| import java.util.Collection; | import java.util.Collection; | ||||||
| import java.util.Comparator; | import java.util.Comparator; | ||||||
|  | import java.util.Objects; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * <p>带缓存功能的{@link SynthesizedAnnotationAttributeProcessor}实现, |  * <p>带缓存功能的{@link SynthesizedAnnotationAttributeProcessor}实现, | ||||||
| @@ -47,16 +47,19 @@ public class CacheableSynthesizedAnnotationAttributeProcessor implements Synthes | |||||||
| 	@Override | 	@Override | ||||||
| 	public <T> T getAttributeValue(String attributeName, Class<T> attributeType, Collection<? extends SynthesizedAnnotation> synthesizedAnnotations) { | 	public <T> T getAttributeValue(String attributeName, Class<T> attributeType, Collection<? extends SynthesizedAnnotation> synthesizedAnnotations) { | ||||||
| 		Object value = valueCaches.get(attributeName, attributeType); | 		Object value = valueCaches.get(attributeName, attributeType); | ||||||
| 		// 此处理论上不可能出现缓存值为nul的情况 | 		if (Objects.isNull(value)) { | ||||||
| 		if (ObjectUtil.isNotNull(value)) { | 			synchronized (valueCaches) { | ||||||
| 			return (T)value; | 				value = valueCaches.get(attributeName, attributeType); | ||||||
|  | 				if (Objects.isNull(value)) { | ||||||
|  | 					value = synthesizedAnnotations.stream() | ||||||
|  | 						.filter(ma -> ma.hasAttribute(attributeName, attributeType)) | ||||||
|  | 						.min(annotationComparator) | ||||||
|  | 						.map(ma -> ma.getAttributeValue(attributeName)) | ||||||
|  | 						.orElse(null); | ||||||
|  | 					valueCaches.put(attributeName, attributeType, value); | ||||||
|  | 				} | ||||||
|  | 			} | ||||||
| 		} | 		} | ||||||
| 		value = synthesizedAnnotations.stream() |  | ||||||
| 			.filter(ma -> ma.hasAttribute(attributeName, attributeType)) |  | ||||||
| 			.min(annotationComparator) |  | ||||||
| 			.map(ma -> ma.getAttributeValue(attributeName)) |  | ||||||
| 			.orElse(null); |  | ||||||
| 		valueCaches.put(attributeName, attributeType, value); |  | ||||||
| 		return (T)value; | 		return (T)value; | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|   | |||||||
| @@ -95,7 +95,7 @@ public class SynthesizedAnnotationProxy implements InvocationHandler { | |||||||
| 	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { | 	public Object invoke(Object proxy, Method method, Object[] args) throws Throwable { | ||||||
| 		return Opt.ofNullable(methods.get(method.getName())) | 		return Opt.ofNullable(methods.get(method.getName())) | ||||||
| 				.map(m -> m.apply(method, args)) | 				.map(m -> m.apply(method, args)) | ||||||
| 				.orElseGet(() -> ReflectUtil.invoke(this, method, args)); | 				.orElseGet(() -> ReflectUtil.invoke(proxy, method, args)); | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	// ========================= 代理方法 ========================= | 	// ========================= 代理方法 ========================= | ||||||
|   | |||||||
| @@ -0,0 +1,64 @@ | |||||||
|  | package cn.hutool.core.annotation; | ||||||
|  |  | ||||||
|  | import org.junit.Assert; | ||||||
|  | import org.junit.Test; | ||||||
|  |  | ||||||
|  | import java.lang.annotation.Annotation; | ||||||
|  | import java.lang.annotation.ElementType; | ||||||
|  | import java.lang.annotation.Retention; | ||||||
|  | import java.lang.annotation.RetentionPolicy; | ||||||
|  | import java.lang.annotation.Target; | ||||||
|  | import java.lang.reflect.Field; | ||||||
|  | import java.util.ArrayList; | ||||||
|  | import java.util.List; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * @author huangchengxing | ||||||
|  |  */ | ||||||
|  | public class TestIssueI8CLBJ { | ||||||
|  |  | ||||||
|  | 	@Test | ||||||
|  | 	public void test() throws NoSuchFieldException { | ||||||
|  | 		Field field = Foo.class.getDeclaredField("name"); | ||||||
|  | 		Assert.assertNotNull(field); | ||||||
|  | 		Annotation[] annotations = field.getDeclaredAnnotations(); | ||||||
|  | 		Assert.assertTrue(annotations.length > 0); | ||||||
|  |  | ||||||
|  | 		TestAnnotation annotation = AnnotationUtil.getSynthesizedAnnotation(TestAnnotation.class, annotations); | ||||||
|  | 		List<Thread> threadList = new ArrayList<>(); | ||||||
|  | 		for (int i = 0; i < 30; i++) { | ||||||
|  | 			Thread thread = new Thread(() -> { | ||||||
|  | 				try { | ||||||
|  | 					String valueFieldName = annotation.valueFieldName(); | ||||||
|  | 					System.out.println("valueFieldName:" + valueFieldName); | ||||||
|  | 				} catch (Exception e) { | ||||||
|  | 					e.printStackTrace(); | ||||||
|  | 				} | ||||||
|  | 			}); | ||||||
|  | 			threadList.add(thread); | ||||||
|  | 			thread.start(); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		try { | ||||||
|  | 			for (Thread thread : threadList) { | ||||||
|  | 				thread.join(); | ||||||
|  | 			} | ||||||
|  | 		} catch (Exception e) { | ||||||
|  | 			e.printStackTrace(); | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	public static class Foo { | ||||||
|  | 		private Integer id; | ||||||
|  | 		@TestAnnotation("name") | ||||||
|  | 		private String name; | ||||||
|  | 	} | ||||||
|  |  | ||||||
|  | 	@Target(ElementType.FIELD) | ||||||
|  | 	@Retention(RetentionPolicy.RUNTIME) | ||||||
|  | 	private @interface TestAnnotation { | ||||||
|  | 		String value() default ""; | ||||||
|  | 		@Alias("value") | ||||||
|  | 		String valueFieldName() default ""; | ||||||
|  | 	} | ||||||
|  | } | ||||||
		Reference in New Issue
	
	Block a user
	 Looly
					Looly