我正在处理一个Java servlet,需要将一个类序列化和反序列化为JSON,然后再反序列化为JSON。为此,我使用了Genson库,但遇到了一个问题。
Genson无法反序列化Date类的实例(java.sql.Date)
我尝试过设置自定义日期格式化程序,但它们似乎不会影响反序列化。
我还尝试使用构建器调用withConverter()插入新的转换器,但我不知道参数是如何工作的。
这是我的构建者的电话
Genson genson = builder.setSkipNull(true).create();我正在序列化的类有一个Date类型的字段
private Date introDate;这是我试图反序列化生成的JSON时出现的堆栈跟踪的一个片段
Caused by: com.owlike.genson.JsonBindingException: Could not access value of property named 'hours' using accessor public int java.sql.Date.getHours() from class java.sql.Date
at com.owlike.genson.reflect.PropertyAccessor.couldNotAccess(PropertyAccessor.java:40)
at com.owlike.genson.reflect.PropertyAccessor$MethodAccessor.access(PropertyAccessor.java:70)
at com.owlike.genson.reflect.PropertyAccessor.serialize(PropertyAccessor.java:24)
at com.owlike.genson.reflect.BeanDescriptor.serialize(BeanDescriptor.java:92)
at com.owlike.genson.convert.NullConverterFactory$NullConverterWrapper.serialize(NullConverterFactory.java:69)
at com.owlike.genson.reflect.PropertyAccessor.serialize(PropertyAccessor.java:27)
... 38 more这里的问题是,像getHours()这样的方法被弃用了,因此调用它们会产生一个IllegalArgumentException。我暂时不知道如何绕过这个问题。
发布于 2019-04-14 06:29:00
您可以使用java.util.Date。这对Genson很有效。
如果我们坚持使用java.sql.Date,那么您可以编写自己的转换器并让Genson使用它。
让我们从一个要服务/去服务的对象开始:
import java.sql.Date;
import lombok.Getter;
import lombok.Setter;
public class CrashTestDummy {
@Getter @Setter private String name;
@Getter @Setter private Date sqlDate;
/** Default no-arg constructor */
public CrashTestDummy() {
}
}然后我们可以告诉Genson使用自定义的转换器。在这种情况下,我将它转换为一个很长的纪元时间,然后再转换回来。您可能决定改用特定的日期格式。
import java.sql.Date;
import java.text.ParseException;
import java.text.SimpleDateFormat;
import org.junit.Test;
import com.owlike.genson.Context;
import com.owlike.genson.Converter;
import com.owlike.genson.Genson;
import com.owlike.genson.GensonBuilder;
import com.owlike.genson.stream.ObjectReader;
import com.owlike.genson.stream.ObjectWriter;
@Test
public void serialiseDate() throws ParseException {
// create a converter for java.sql.Date
Converter<Date> converter = new Converter<Date>() {
@Override
public void serialize(Date obj, ObjectWriter writer, Context ctx) throws Exception {
// java.sql.Date doesn't support any time fields, so we can just focus on y, M and d
if(obj == null) {
writer.writeNull();
return;
}
writer.writeValue( obj.getTime() );
}
@Override
public Date deserialize(ObjectReader reader, Context ctx) {
Long value = reader.valueAsLong();
Date sqlDate = new java.sql.Date( value );
return sqlDate;
}
};
// Build a new Genson object with our converter
Genson genson = new GensonBuilder()
.setSkipNull(true)
.withConverter(converter, java.sql.Date.class)
.create();
// make an important SQL date for testing
java.util.Date utilDate = new SimpleDateFormat("dd MMM yyyy").parse("30 July 1966");
Date sqlDate = new Date(utilDate.getTime() );
// Make something to serialise
CrashTestDummy original = new CrashTestDummy();
original.setName( "Alfa Khrisna" );
original.setSqlDate( sqlDate );
// Call Genson as usual
String json = genson.serialize( original );
System.out.println( json );
// Deserialise as usual; for brevity I'm comparing dates in millis since epoch.
CrashTestDummy clone = genson.deserialize(json, CrashTestDummy.class);
assertEquals(new SimpleDateFormat("dd MMM yyyy").parse("30 July 1966").getTime(), clone.getSqlDate().getTime());
assertEquals("Alfa Khrisna", clone.getName());
}https://stackoverflow.com/questions/55559276
复制相似问题