private void method(){ InputStream in; try { Yaml yaml = new Yaml(); in = this.getClass().getResourceAsStream("/ratelimiter-rule.yaml"); if (in != null) { RuleConfig ruleConfig = yaml.loadAs(in, RuleConfig.class); System.out.println(ruleConfig.toString()); Object obj =yaml.load(in); System.out.println(obj); } } catch (Exception e) { e.printStackTrace(); } }
如上为读取yaml文件转java对象代码;
抛出异常:Can't construct a java object for tag:yaml.org,2002:com.jk.RuleConfig; exception=java.lang.InstantiationException: NoSuchMethodException:com.jk.RuleConfig.<init>();
java.lang.InstantiationException: NoSuchMethodException:com.jk.RuleConfig.<init>()
研究发现:RuleConfig类中的有参构造覆盖了无参构造导致的,添加无参构造方法即可。
回忆发现,其实在很多将数据转换为java对象时都会需要无参构造方法,比如通过数据库的数据转换成Entity对象,没有无参构造也会有相似的问题。
还有一个问题:配置文件如下,发现,yaml文件如果是 configs ,解析就可以,其他字段就不行,
configs: - appId: app-1 limits: - api: /v1/user limit: 100 unit: 60 - api: /v1/order limit: 50
最开始以为和RuleConfig类中的成员变量是一致的,验证发现,就是成员变量换个其他名字,也是可以的。故不知问题所在。附上错误异常,是将配置文件中configs改为configs2的异常。
Can't construct a java object for tag:yaml.org,2002:com.jk.RuleConfig; exception=null
in 'reader', line 1, column 1:
configs2:
查看源码发现,其实正常情况下configs 和成员变量名称保持一致是可以的,因为成员变量的set方法命名就是根据set+成员变量名称来定义的。而我这里不行是因为我只修改了成员变量没有修改set方法,可以说set方法不是常规,在通过yaml解析为RuleConfig对象时是通过反射获取类中的方法,解析方法名来判断和yaml中定义的变量(也就是configs )是否一致,如果一致还会判断此方法是不是writable的,表示这个方法是不是写方法,如果是则校验合法可以构建对象,如果不是则校验失败跑出异常。
// public class PropertyUtils 类,获取类的成员方法,解析后properties 中。
protected Map<String, Property> getPropertiesMap(Class<?> type, BeanAccess bAccess) { if (propertiesCache.containsKey(type)) { return propertiesCache.get(type); } Map<String, Property> properties = new LinkedHashMap<String, Property>(); boolean inaccessableFieldsExist = false; switch (bAccess) { case FIELD: // ... break; default: // add JavaBean properties try { for (PropertyDescriptor property : Introspector.getBeanInfo(type) .getPropertyDescriptors()) { Method readMethod = property.getReadMethod(); if ((readMethod == null || !readMethod.getName().equals("getClass")) && !isTransient(property)) { properties.put(property.getName(), new MethodProperty(property)); } } } catch (IntrospectionException e) { throw new YAMLException(e); } // add public fields // ..... } if (properties.isEmpty() && inaccessableFieldsExist) { throw new YAMLException("No JavaBean properties found in " + type.getName()); } propertiesCache.put(type, properties); return properties; }
public class Constructor extends SafeConstructor{ }中的
protected Object constructJavaBean2ndStep(MappingNode node, Object object)的方法片段。
try { TypeDescription memberDescription = typeDefinitions.get(beanType); Property property = memberDescription == null ? getProperty(beanType, key) : memberDescription.getProperty(key); // 此处就是判读之前放入properties 中的方法是不是writable if (!property.isWritable()) { throw new YAMLException("No writable property '" + key + "' on class: " + beanType.getName()); }