提交 f613c3b7 编写于 作者: Y yukon

[ROCKETMQ-259]Fix too many reflection calls when decode remoting command header

上级 f091203a
...@@ -43,7 +43,7 @@ public class RemotingCommand { ...@@ -43,7 +43,7 @@ public class RemotingCommand {
private static final Map<Class, String> CANONICAL_NAME_CACHE = new HashMap<Class, String>(); private static final Map<Class, String> CANONICAL_NAME_CACHE = new HashMap<Class, String>();
// 1, Oneway // 1, Oneway
// 1, RESPONSE_COMMAND // 1, RESPONSE_COMMAND
private static final Map<Field, Annotation> NOT_NULL_ANNOTATION_CACHE = new HashMap<Field, Annotation>(); private static final Map<Field, Boolean> NULLABLE_FIELD_CACHE = new HashMap<Field, Boolean>();
private static final String STRING_CANONICAL_NAME = String.class.getCanonicalName(); private static final String STRING_CANONICAL_NAME = String.class.getCanonicalName();
private static final String DOUBLE_CANONICAL_NAME_1 = Double.class.getCanonicalName(); private static final String DOUBLE_CANONICAL_NAME_1 = Double.class.getCanonicalName();
private static final String DOUBLE_CANONICAL_NAME_2 = double.class.getCanonicalName(); private static final String DOUBLE_CANONICAL_NAME_2 = double.class.getCanonicalName();
...@@ -252,11 +252,9 @@ public class RemotingCommand { ...@@ -252,11 +252,9 @@ public class RemotingCommand {
try { try {
String value = this.extFields.get(fieldName); String value = this.extFields.get(fieldName);
if (null == value) { if (null == value) {
Annotation annotation = getNotNullAnnotation(field); if (!isFieldNullable(field)) {
if (annotation != null) {
throw new RemotingCommandException("the custom field <" + fieldName + "> is null"); throw new RemotingCommandException("the custom field <" + fieldName + "> is null");
} }
continue; continue;
} }
...@@ -305,16 +303,14 @@ public class RemotingCommand { ...@@ -305,16 +303,14 @@ public class RemotingCommand {
return field; return field;
} }
private Annotation getNotNullAnnotation(Field field) { private boolean isFieldNullable(Field field) {
Annotation annotation = NOT_NULL_ANNOTATION_CACHE.get(field); if (!NULLABLE_FIELD_CACHE.containsKey(field)) {
Annotation annotation = field.getAnnotation(CFNotNull.class);
if (annotation == null) { synchronized (NULLABLE_FIELD_CACHE) {
annotation = field.getAnnotation(CFNotNull.class); NULLABLE_FIELD_CACHE.put(field, annotation == null);
synchronized (NOT_NULL_ANNOTATION_CACHE) {
NOT_NULL_ANNOTATION_CACHE.put(field, annotation);
} }
} }
return annotation; return NULLABLE_FIELD_CACHE.get(field);
} }
private String getCanonicalName(Class clazz) { private String getCanonicalName(Class clazz) {
......
...@@ -16,8 +16,11 @@ ...@@ -16,8 +16,11 @@
*/ */
package org.apache.rocketmq.remoting.protocol; package org.apache.rocketmq.remoting.protocol;
import java.lang.reflect.Field;
import java.lang.reflect.Method;
import java.nio.ByteBuffer; import java.nio.ByteBuffer;
import org.apache.rocketmq.remoting.CommandCustomHeader; import org.apache.rocketmq.remoting.CommandCustomHeader;
import org.apache.rocketmq.remoting.annotation.CFNotNull;
import org.apache.rocketmq.remoting.exception.RemotingCommandException; import org.apache.rocketmq.remoting.exception.RemotingCommandException;
import org.junit.Test; import org.junit.Test;
...@@ -179,6 +182,32 @@ public class RemotingCommandTest { ...@@ -179,6 +182,32 @@ public class RemotingCommandTest {
assertThat(((ExtFieldsHeader) decodedHeader).isBooleanValue()).isEqualTo(true); assertThat(((ExtFieldsHeader) decodedHeader).isBooleanValue()).isEqualTo(true);
assertThat(((ExtFieldsHeader) decodedHeader).getDoubleValue()).isBetween(0.617, 0.619); assertThat(((ExtFieldsHeader) decodedHeader).getDoubleValue()).isBetween(0.617, 0.619);
} }
@Test
public void testNotNullField() throws Exception {
RemotingCommand remotingCommand = new RemotingCommand();
Method method = RemotingCommand.class.getDeclaredMethod("isFieldNullable", Field.class);
method.setAccessible(true);
Field nullString = FieldTestClass.class.getDeclaredField("nullString");
assertThat(method.invoke(remotingCommand, nullString)).isEqualTo(false);
Field nullableString = FieldTestClass.class.getDeclaredField("nullable");
assertThat(method.invoke(remotingCommand, nullableString)).isEqualTo(true);
Field value = FieldTestClass.class.getDeclaredField("value");
assertThat(method.invoke(remotingCommand, value)).isEqualTo(false);
}
}
class FieldTestClass {
@CFNotNull
String nullString = null;
String nullable = null;
@CFNotNull
String value = "NotNull";
} }
class SampleCommandCustomHeader implements CommandCustomHeader { class SampleCommandCustomHeader implements CommandCustomHeader {
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册