如何配置Genson以反序列化日期?

时间:2019-04-07 13:01:23

标签: java json date genson

我正在研究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。我暂时不知道该如何解决。

1 个答案:

答案 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());
    }