Класс Swing DefaultMutableTreeNode
представляет собой древовидную структуру данных, которая содержит экземпляры этого же типа как children
, так и parent
.Вот почему стандартный сериализатор Gson столкнулся с бесконечной рекурсией и, следовательно, выкинул StackOverflowError
.
. Чтобы решить эту проблему, вам нужно настроить Gson
с более умным JsonSerializer
, специально созданным дляпреобразование DefaultMutableTreeNode
в JSON.В качестве бонуса вы также можете предоставить JsonDeserializer
для преобразования такого JSON обратно в DefaultMutableTreeNode
.
. Для этого создайте свой экземпляр Gson
, а не просто new Gson()
, но
Gson gson = new GsonBuilder()
.registerTypeAdapter(DefaultMutableTreeNode.class, new DefaultMutableTreeNodeSerializer())
.registerTypeAdapter(DefaultMutableTreeNode.class, new DefaultMutableTreeNodeDeserializer())
.setPrettyPrinting()
.create();
DefaultMutableTreeNodeSerializer
ниже отвечает за преобразование DefaultMutableTreeNode
в JSON.Он преобразует свои свойства allowsChildren
, userObject
и children
в JSON.Обратите внимание, что он не преобразует свойство parent
в JSON, потому что выполнение этого приведет к повторной бесконечной рекурсии.
public class DefaultMutableTreeNodeSerializer implements JsonSerializer<DefaultMutableTreeNode> {
@Override
public JsonElement serialize(DefaultMutableTreeNode src, Type typeOfSrc, JsonSerializationContext context) {
JsonObject jsonObject = new JsonObject();
jsonObject.addProperty("allowsChildren", src.getAllowsChildren());
jsonObject.add("userObject", context.serialize(src.getUserObject()));
if (src.getChildCount() > 0) {
jsonObject.add("children", context.serialize(Collections.list(src.children())));
}
return jsonObject;
}
}
Для тестирования давайте сериализуем корневой узел образца JTree
в JSON,и затем снова десериализовать его.
JTree tree = new JTree(); // create a sample tree
Object topNode = tree.getModel().getRoot(); // a DefaultMutableTreeNode
String jsonString = gson.toJson(topNode);
System.out.println(jsonString);
DefaultMutableTreeNode topNode2 = gson.fromJson(jsonString, DefaultMutableTreeNode.class);
Он генерирует следующий вывод JSON:
{
"allowsChildren": true,
"userObject": "JTree",
"children": [
{
"allowsChildren": true,
"userObject": "colors",
"children": [
{
"allowsChildren": true,
"userObject": "blue"
},
{
"allowsChildren": true,
"userObject": "violet"
},
{
"allowsChildren": true,
"userObject": "red"
},
{
"allowsChildren": true,
"userObject": "yellow"
}
]
},
{
"allowsChildren": true,
"userObject": "sports",
"children": [
{
"allowsChildren": true,
"userObject": "basketball"
},
{
"allowsChildren": true,
"userObject": "soccer"
},
{
"allowsChildren": true,
"userObject": "football"
},
{
"allowsChildren": true,
"userObject": "hockey"
}
]
},
{
"allowsChildren": true,
"userObject": "food",
"children": [
{
"allowsChildren": true,
"userObject": "hot dogs"
},
{
"allowsChildren": true,
"userObject": "pizza"
},
{
"allowsChildren": true,
"userObject": "ravioli"
},
{
"allowsChildren": true,
"userObject": "bananas"
}
]
}
]
}
DefaultMutableTreeNodeDeserializer
ниже отвечает за преобразование JSON обратно в DefaultMutableTreeNode
.
Он использует ту же идею, что и десериализатор из Как сериализовать / десериализовать DefaultMutableTreeNode с Джексоном? .DefaultMutableTreeNode
не очень похож на POJO и поэтому не очень хорошо работает вместе с Gson.Следовательно, он использует хорошо работающий вспомогательный класс POJO
(со свойствами allowsChildren
, userObject
и children
) и позволяет Gson десериализовать содержимое JSON в этот класс.Затем объект POJO
(и его POJO
потомков) преобразуется в объект DefaultMutableTreeNode
(с DefaultMutableTreeNode
потомками).
public class DefaultMutableTreeNodeDeserializer implements JsonDeserializer<DefaultMutableTreeNode> {
@Override
public DefaultMutableTreeNode deserialize(JsonElement json, Type type, JsonDeserializationContext context) {
return context.<POJO>deserialize(json, POJO.class).toDefaultMutableTreeNode();
}
private static class POJO {
private boolean allowsChildren;
private Object userObject;
private List<POJO> children;
// no need for: POJO parent
public DefaultMutableTreeNode toDefaultMutableTreeNode() {
DefaultMutableTreeNode node = new DefaultMutableTreeNode();
node.setAllowsChildren(allowsChildren);
node.setUserObject(userObject);
if (children != null) {
for (POJO child : children) {
node.add(child.toDefaultMutableTreeNode()); // recursion!
// this did also set the parent of the child-node
}
}
return node;
}
// Following setters needed by Gson's deserialization:
public void setAllowsChildren(boolean allowsChildren) {
this.allowsChildren = allowsChildren;
}
public void setUserObject(Object userObject) {
this.userObject = userObject;
}
public void setChildren(List<POJO> children) {
this.children = children;
}
}
}