从未调用Date的自定义gson反序列化程序

4uqofj5v  于 2022-11-06  发布在  其他
关注(0)|答案(2)|浏览(180)

据我所知,gson不会自动将java.util.Date对象序列化和反序列化为ISO字符串,例如“yyyy-MM-ddTHH:mm:ssZ”或“2014-04- 15 T18:22:00-05:00”。因此,为了让我在客户机(使用带有gson的Retrofit)和服务器之间正确地传递日期,我需要为gson指定DateFormat。下面是我所做的工作:

// code defining the creation of a RestAdapter
// ...
new GsonBuilder()
    .setDateFormat("yyyy-MM-dd'T'HH:mm:ssZ")
    .create()

添加.setDateFormat行就足以让gson正确地将时间戳字符串反序列化为Date对象。但是,它并没有将Date对象序列化为时间戳字符串。因此,我假设我必须创建一个自定义序列化器,如下所示:

// code defining the creation of a RestAdapter
// ...
new GsonBuilder()
    .setDateFormat("yyyy-MM-dd'T'HH:mm:ssZ")
    .registerTypeAdapter(Date.class, new DateSerializer())
    .create()

和DateSerializer类别:

class DateSerializer implements JsonSerializer<Date> {
    @Override
    public JsonElement serialize(Date arg0, Type arg1, JsonSerializationContext arg2) {
        SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ssZ", Locale.US);
        return new JsonPrimitive(df.format(arg0));
    }
}

不幸的是,serialize函数被忽略了。相反,gson将日期格式化为字符串,如“Tues Mar 15 18:22:00 EST 2014”。因此,为了测试,我尝试将serialize函数替换为:

public JsonElement serialize(Date arg0, Type arg1, JsonSerializationContext arg2) {
    throw new RuntimeException("Serialize function called!");
}

但是,当然不会抛出RuntimeException。
有人知道为什么我的serialize函数被忽略了吗?我想我在什么地方读到过,对于某些类型,如果有一个为超类定义的registerTypeAdapter,它将被忽略,但是由于这是java.util.Date,如果这是问题所在,我会感到困惑。我可能只是在做一些愚蠢的事情,但是我可能对Date或gson不够了解,无法意识到这一点。
编辑:提供了有关以下代码的更多上下文:

我的应用程序.java

public class MyApplication extends Application {
    public static RestAdapter restAdapter;
    public static The1Api the1Api;

    public static void createRestAdapter(String server_url){
        // enable cookies
        CookieManager cookieManager = new CookieManager();
        cookieManager.setCookiePolicy(CookiePolicy.ACCEPT_ALL);
        CookieHandler.setDefault(cookieManager);

        // create rest adapter
        restAdapter = new RestAdapter.Builder()
            .setEndpoint(server_url)
            .setConverter(new GsonConverter(new GsonBuilder()
                    .setDateFormat("yyyy-MM-dd'T'HH:mm:ssZ")
                    .registerTypeAdapter(Date.class, new DateSerializer())
                    .create()))
            .setLogLevel(LogLevel.FULL)
            .setLog(new ResponseInterceptor())
            .build();

        // create API
        the1Api = restAdapter.create(The1Api.class);
    }
}

1Api.java语言

public interface The1Api {
    /* Chat */

    public class PublicMessage {
        String from_user;
        String message;
        Date time;
        Integer microsecond;
        Boolean in_1_percent;
    }
    public class PublicMessageList {
        Integer last_message_id;
        ArrayList<PublicMessage> messages;
    }

    @GET("/chat/get_public_messages/")
    public PublicMessageList getPublicMessages(
            @Query("last_message_id") Integer last_message_id, // optional
            @Query("since") Date since, // optional
            @Query("max") Integer max // optional
            );   

    // ...
}

登录活动.java

public class LoginActivity extends Activity {
    // ...

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        Crashlytics.start(this);
        MyApplication.createRestAdapter(getString(R.string.server_url));
        setContentView(R.layout.activity_login);
    }

    @Override
    protected void onPostCreate(Bundle savedInstanceState){
        super.onPostCreate(savedInstanceState);

        Thread thread = new Thread(){
            @Override
            public void run(){
                ArrayList<The1Api.PublicMessage> publicMessages = MyApplication.the1Api.getPublicMessages(null, null, null).messages;
                for (The1Api.PublicMessage m : publicMessages){
                    Log.d("The1", "[" + m.time.toString() + "] " + m.from_user + ": " + m.message);
                }
                // when the following line gets executed, my server receives a request including the date below,
                // but the server does not understand the format of the date because it does not get serialized properly
                MyApplication.the1Api.getPublicMessages(null, new Date(1000000000), null);
            }
        };
        thread.start();
    }

    // ...    
}

日期序列化程序.java

class DateSerializer implements JsonSerializer<Date> {
    @Override
    public JsonElement serialize(Date arg0, Type arg1, JsonSerializationContext arg2) {
        SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ssZ", Locale.US);
        return new JsonPrimitive(df.format(arg0));
    }
}

编辑2:目前还没有解决方案,但作为一种解决方案,你可以在发送之前手动将日期转换为其他格式。在Kalel建议将其转换为字符串的注解中,我将其转换为长(自UNIX纪元以来的秒数)。

kse8i1jr

kse8i1jr1#

你还能重现这个问题吗?我看到Gson正确地序列化和反序列化。这可能是一个Retrofit问题(我以前没有用过它),但是查看GsonConverter的文档,我看不出它除了委托给Gson对象中提供的对象之外还有什么别的原因。
这是一个SSCCE:

public class GsonDateDemo {
  public static void main(String[] args) {
    Gson gson = new GsonBuilder()
        .setDateFormat("yyyy-MM-dd'T'HH:mm:ssZ")
        .registerTypeAdapter(Date.class, new DateSerializer())
        .create();
    Date d = new Date(1438806977000L);

    System.out.println("Serializing "+d);
    String json = gson.toJson(d);
    System.out.println("JSON: "+json);
    Date d2 = gson.fromJson(json, Date.class);
    System.out.println("Deserialized: "+d2);
    System.out.println("Equal? "+d.equals(d2));
  }

  static class DateSerializer implements JsonSerializer<Date> {
    @Override
    public JsonElement serialize(Date arg0, Type arg1, JsonSerializationContext arg2) {
      System.out.println("Serializer received "+arg0);
      SimpleDateFormat df = new SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ssZ", Locale.US);
      return new JsonPrimitive(df.format(arg0));
    }
  }
}

这将输出:

Calling: local.GsonDateDemo.main[]
Serializing Wed Aug 05 16:36:17 EDT 2015
Serializer received Wed Aug 05 16:36:17 EDT 2015
JSON: "2015-08-05T16:36:17-0400"
Deserialized: Wed Aug 05 16:36:17 EDT 2015
Equal? true

如果可以,请修改此SSCCE以使用Retrofit,并查看是否可以通过这种方式复制故障。

643ylb08

643ylb082#

gson中的问题,当我们尝试在调用setDateFormat后反序列化时,自定义的DateDesializer没有调用。
因此,当您进行反序列化时,不要调用setDateFormat;)

相关问题