我正在研究Java servlet,我需要将一个类序列化和反序列化为JSON并返回。为此,我正在使用Genson库,但是遇到了麻烦。
Genson无法反序列化Date类(java.sql.Date)的实例
我尝试设置自定义日期格式,但它们似乎不会影响反序列化。
我也尝试过使用带有conConverter()的构建器调用来插入新的转换器,但是我无法弄清楚参数是如何工作的。
这是我的建设者电话
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。我暂时不知道该如何解决。
答案 0 :(得分:1)
您可以使用java.util.Date。可以和Genson一起使用。
如果我们坚持使用java.sql.Date,那么您可以编写自己的转换器并让Genson使用它。
让我们从一个Ser / deser对象开始:
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());
}