3

我知道如何序列化空对象。但是如果我有一个对象并且在另一个为空的对象内部怎么办?

例如

我的第一堂课:

public class Invoice {

   private Adresse adresse;
   private Double betrag;
   private Double Ust;
   private String zweck;

}

我的第二堂课:

public class Adresse {
   private String name;
   private Ort ort;
}

还有更多的子类...

如果我这样做:

Gson gson = new GsonBuilder().serializeNulls().create();

我只得到这个:

{"adresse":null,"betrag":null,"Ust":null,"zweck":null}

而不是这个:

{"adresse":{"name":null,"ort"{"plz":null,"name":null}},"betrag":null,"Ust":null,"zweck":null}

应该有一种自动化的方法。总会有其他子类。因此,手动方法是行不通的。

4

2 回答 2

1

我让它工作了!使用 Java 反射!

这里的代码:

  1. userDefined()检查一个类是用户定义的还是原始的,可以根据需要进行自定义。

    public boolean isUserDefined(Class o) {
      if (o.isAssignableFrom(String.class)) {
        return false;
      }
      if (o.isAssignableFrom(Double.class)) {
        return false;
      }
      if (o.isAssignableFrom(Integer.class)) {
        return false;
      }
      if (o.isAssignableFrom(Boolean.class)) {
        return false;
      }
      if (o.isAssignableFrom(Short.class)) {
        return false;
      }
      if (o.isAssignableFrom(Float.class)) {
        return false;
      }
      if (o.isAssignableFrom(Long.class)) {
        return false;
      }
      return true;
    }
    
  2. getSetterMethod()返回字段的设置方法。

    private static Method getSetterMethod(Object o, Field f) {       
        for (Method method : o.getClass().getMethods()) {
            String fieldname = "set" + f.getName();
            String methodname = method.getName().toLowerCase();
            if (fieldname.equals(methodname)) {
                return method;
            }
        }
        return null;
    }
    
  3. 最后,invokeSetters()调用用户定义的对象 setter 方法和子类的方法!是的。

        private void invokeSetters(Object o) throws IllegalArgumentException, IllegalAccessException, ClassNotFoundException, InstantiationException, NoSuchMethodException, InvocationTargetException {
        System.out.println("INVOKING THE OBJECT " + o.getClass().toString());
          for (Field f : o.getClass().getDeclaredFields()) {
            System.out.println("NOW IN THE FIELD: " + f.getName());
            f.setAccessible(true);
            Object obj = null;
            if (isUserDefined(f.getType()) && f.get(o) == null) {
                System.out.println("FIELD is USER DEFINED AND NULL");
                Class c = Class.forName(f.getType().getName());
                System.out.println("CLASS IS " + c.getName());
                obj = c.getConstructor().newInstance();
                System.out.println("GOING INSIDE " + obj.toString() + " NOW.");
                invokeSetters(obj);
            }
            if (obj != null) {
                Method setter = getSetterMethod(o, f);
                System.out.println("IM IN PARENT OBJECT " + o + " AND INVOKING METHOD " + setter.getName() + " WITH PARAMETER " + obj.toString());
                setter.invoke(o, obj);
                System.out.println("METHOD WAS INVOKED");
            }
          }
        }
    
于 2012-11-22T02:49:47.373 回答
0

如果你想adresse拥有这些属性,那么你需要实例adresse化为:

    public class Invoice implements Serializable{
        private Adresse adresse = new Adresse();
        private Double betrag;
        private Double Ust;
        private String zweck;
    }
于 2012-11-19T03:25:38.403 回答