![](/img/trans.png)
[英]Java.lang.NoClassDefFoundError: com/fasterxml/jackson/databind/exc/InvalidDefinitionException
[英]Java InvalidDefinitionException when serializing object with jackson databind
我正在尝试使用Jackson的ObjectMapper将以下Player对象编写为String。
package models.Game;
import models.Game.Enums.SnowballState;
import javafx.scene.paint.Color;
import javafx.scene.shape.Circle;
import java.util.ArrayList;
import java.util.List;
public class Player {
private Circle circle;
private String name;
private Color color;
private int points = 0;
public int getLives() {
return lives;
}
private int lives = 3;
private List<Snowball> snowballs;
private Circle oldCircle;
private int stepSize = 10;
public Player(String name, Color color) {
this.name = name;
circle = new Circle();
oldCircle = new Circle();
this.color = color;
snowballs = new ArrayList<>();
snowballs.add(new Snowball(this));
snowballs.add(new Snowball(this));
snowballs.add(new Snowball(this));
}
public Player() {
}
private void removeLife() {
this.lives--;
}
public int getHit() {
removeLife();
return getLives();
}
public int shotSuccess() {
points+= 50;
return points;
}
public int getSnowballAmount() {
int balls = 0;
for (Snowball ball : snowballs) {
if (ball.getState() == SnowballState.CREATED) {
balls++;
}
}
return balls;
}
public List<Snowball> getSnowballs() {
return snowballs;
}
public Snowball getNextSnowball() {
for (Snowball ball : snowballs) {
if (ball.getState() == SnowballState.CREATED) {
return ball;
}
}
return null;
}
public void createSnowball() {
if (getSnowballAmount() < 3) {
snowballs.add(new Snowball(this));
}
}
public Color getColor() {
return this.color;
}
public Circle getCircle() {
return this.circle;
}
public void moveLeft() {
saveOld();
circle.setTranslateX(circle.getTranslateX() - stepSize);
}
public void moveRight() {
saveOld();
circle.setTranslateX(circle.getTranslateX() + stepSize);
}
public void moveUp() {
saveOld();
circle.setTranslateY(circle.getTranslateY() - stepSize);
}
public void moveDown() {
saveOld();
circle.setTranslateY(circle.getTranslateY() + stepSize);
}
public void undo() {
circle.setTranslateX(oldCircle.getTranslateX());
circle.setTranslateY(oldCircle.getTranslateY());
}
private void saveOld() {
oldCircle.setTranslateX(circle.getTranslateX());
oldCircle.setTranslateY(circle.getTranslateY());
}
public Snowball shootSnowball(Snowball ball, double mouseX, double mouseY) {
double polarDirection = Math.atan2(mouseY - circle.getTranslateY(), mouseX - circle.getTranslateX() + 50);
ball.setState(SnowballState.ALIVE);
ball.setDirection(polarDirection);
ball.getCircle().setTranslateX(circle.getTranslateX() + 50);
ball.getCircle().setTranslateY(circle.getTranslateY());
return ball;
}
我正在使用以下命令执行此操作:
String json = null;
try {
json = objectMapper.writeValueAsString(instanceOfPlayerClass);
} catch (JsonProcessingException e) {
e.printStackTrace();
}
不幸的是,我收到以下相关的错误消息:
com.fasterxml.jackson.databind.exc.InvalidDefinitionException:类型
com.sun.javafx.scene.NodeEventDispatcher
类型定义无效:无法为[simple type,class com.sun.javafx.scene.NodeEventDispatcher]构造BeanSerializer:(java .lang.reflect.InaccessibleObjectException)无法使公共最终com.sun.javafx.event.BasicEventDispatcher com.sun.javafx.event.BasicEventDispatcher.getPreviousDispatcher()可访问:模块javafx.base不“导出com.sun.javafx。 event“to module com.fasterxml.jackson.databind(通过参考链:models.communication.Websockets.ConnectionSubmitModel [”player“] - > models.Game.Player [”circle“] - > javafx.scene.shape.Circle [ “父”] - > javafx.scene.layout.GridPane [ “父”] - > javafx.scene.layout.AnchorPane [ “此事件”])
就像错误说它与JavaFx没有导出某种依赖关系有关,但由于我不能控制JavaFx,我不确定如何解决这个问题。
您正在尝试存储Circle
类,它是一个JavaFX类,它实际上不是一个数据类(它是一个UI元素),具有许多属性(如半径,粗细,颜色,填充,边框等)。 因此,它以各种方式与JavaFX系统捆绑在一起,并且不能很好地存储。
相反,只需将您想要的信息存储在您自己的简单类中,该类具有在您阅读时再次创建Circle
对象所需的信息。
一般来说Jackson
在POJO
课程中表现最好。 如果要序列化业务对象,可能会发生许多意外错误。 可能最好的解决方案是创建代表Player
和Snowball
状态的新模型类。 像PlayerState
和SnowballState
类的东西。 这两个类应遵循POJO
规则: getters
, setters
, no-arg constructor
等。当您需要将状态保存到JSON
您可以将业务模型转换为状态模型和序列化状态模型 。 当您需要反序列化JSON
,需要将其反序列化为状态模型 ,然后将其转换为业务模型 。 对于JavaFX
类,如果需要,您需要实现自定义序列化器和解串器。 他们也不是常规的POJO
课程,需要特殊待遇。
让我们实现两个序列化器和一个解串器:
class CircleJsonSerializer extends JsonSerializer<Circle> {
@Override
public void serialize(Circle value, JsonGenerator gen, SerializerProvider serializers) throws IOException {
gen.writeStartObject();
gen.writeNumberField("radius", value.getRadius());
gen.writeNumberField("centerX", value.getCenterX());
gen.writeNumberField("centerY", value.getCenterY());
gen.writeEndObject();
}
}
class CircleJsonDeserializer extends JsonDeserializer<Circle> {
@Override
public Circle deserialize(JsonParser p, DeserializationContext ctxt) throws IOException {
TreeNode node = p.readValueAsTree();
NumericNode radius = (NumericNode) node.get("radius");
NumericNode centerX = (NumericNode) node.get("centerX");
NumericNode centerY = (NumericNode) node.get("centerY");
return new Circle(centerX.doubleValue(), centerY.doubleValue(), radius.doubleValue());
}
}
class ColorJsonDeserializer extends JsonDeserializer<Color> {
@Override
public Color deserialize(JsonParser p, DeserializationContext ctxt) throws IOException {
TreeNode node = p.readValueAsTree();
NumericNode red = (NumericNode) node.get("red");
NumericNode green = (NumericNode) node.get("green");
NumericNode blue = (NumericNode) node.get("blue");
NumericNode opacity = (NumericNode) node.get("opacity");
return Color.color(red.doubleValue(), green.doubleValue(), blue.doubleValue(), opacity.doubleValue());
}
}
你可以使用它们如下:
import com.fasterxml.jackson.annotation.JsonIgnore;
import com.fasterxml.jackson.core.JsonGenerator;
import com.fasterxml.jackson.core.JsonParser;
import com.fasterxml.jackson.core.TreeNode;
import com.fasterxml.jackson.databind.DeserializationContext;
import com.fasterxml.jackson.databind.JsonDeserializer;
import com.fasterxml.jackson.databind.JsonSerializer;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.SerializationFeature;
import com.fasterxml.jackson.databind.SerializerProvider;
import com.fasterxml.jackson.databind.module.SimpleModule;
import com.fasterxml.jackson.databind.node.NumericNode;
import javafx.scene.paint.Color;
import javafx.scene.shape.Circle;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;
public class JsonApp {
public static void main(String[] args) throws Exception {
Player player = new Player("N1", Color.BLUE);
SimpleModule javafxModule = new SimpleModule();
javafxModule.addSerializer(Circle.class, new CircleJsonSerializer());
javafxModule.addDeserializer(Circle.class, new CircleJsonDeserializer());
javafxModule.addDeserializer(Color.class, new ColorJsonDeserializer());
ObjectMapper mapper = new ObjectMapper();
mapper.registerModule(javafxModule);
mapper.enable(SerializationFeature.INDENT_OUTPUT);
String json = mapper.writeValueAsString(player);
System.out.println(json);
System.out.println(mapper.readValue(json, Player.class));
}
}
上面的代码打印:
{
"circle" : {
"radius" : 1.0,
"centerX" : 0.0,
"centerY" : 0.0
},
"color" : {
"red" : 0.0,
"green" : 0.0,
"blue" : 1.0,
"opacity" : 1.0,
"opaque" : true,
"hue" : 240.0,
"saturation" : 1.0,
"brightness" : 1.0
},
"lives" : 3,
"snowballs" : [ {
"state" : "CREATED",
"direction" : 0.0,
"circle" : null
}, {
"state" : "CREATED",
"direction" : 0.0,
"circle" : null
}, {
"state" : "CREATED",
"direction" : 0.0,
"circle" : null
} ]
}
//ToString
Player{circle=Circle[centerX=0.0, centerY=0.0, radius=1.0, fill=0x000000ff], name='null', color=0x0000ffff, points=0, lives=3, snowballs=[Snowball{player=null, state=CREATED, direction=0.0, circle=null}, Snowball{player=null, state=CREATED, direction=0.0, circle=null}, Snowball{player=null, state=CREATED, direction=0.0, circle=null}], oldCircle=null, stepSize=10}
正如您所看到的,我们可以序列化和反序列化Player
类,但它需要许多额外的工作。 对于执行业务逻辑的每个getter
方法,我也忽略了它们,如下所示:
@JsonIgnore
public int getHit() {
removeLife();
return getLives();
}
还有一个提示: getHint
方法有副作用。 它消除了生命 - 无论它意味着什么。 这通常是一种不好的做法,但这个问题与命名无关。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.