web-dev-qa-db-ja.com

Jacksonを使用したMap <Date、String>のシリアル化

ジャクソンで地図をシリアル化したい。日付は、他のすべての日付と同様に、タイムスタンプとしてシリアル化する必要があります。

次のコードは、タイムスタンプの代わりに「Tue Mar 11 00:00:00 CET 1952」(Date.toString())の形式でキーをレンダリングします。

Map<Date, String> myMap = new HashMap<Date, String>();
...
ObjectMapper.writeValue(myMap)

これはタイプ消去のためであり、ジャクソンは実行時にキーが日付であることを認識していません。しかし、TypeReferenceをwriteValueメソッドに渡す方法が見つかりませんでした。

私の望ましい動作を実現する簡単な方法はありますか、またはすべてのキーは常にジャクソンによって文字列としてレンダリングされますか?

ヒントをありがとう。

30
Florian Gutmann

デフォルトのマップキーシリアライザはStdKeySerializerであり、これを行うだけです。

String keyStr = (value.getClass() == String.class) ? ((String) value) : value.toString();
jgen.writeFieldName(keyStr);

SimpleModule機能 を使用し、addKeySerializerメソッドを使用してカスタムキーシリアライザーを指定できます。


そして、それを実現する方法を以下に示します。

import Java.io.IOException;
import Java.util.Date;
import Java.util.HashMap;
import Java.util.Map;

import org.codehaus.jackson.JsonGenerator;
import org.codehaus.jackson.JsonProcessingException;
import org.codehaus.jackson.Version;
import org.codehaus.jackson.map.JsonSerializer;
import org.codehaus.jackson.map.ObjectMapper;
import org.codehaus.jackson.map.ObjectWriter;
import org.codehaus.jackson.map.SerializerProvider;
import org.codehaus.jackson.map.module.SimpleModule;
import org.codehaus.jackson.map.type.MapType;
import org.codehaus.jackson.map.type.TypeFactory;

public class CustomKeySerializerDemo
{
  public static void main(String[] args) throws Exception
  {
    Map<Date, String> myMap = new HashMap<Date, String>();
    myMap.put(new Date(), "now");
    Thread.sleep(100);
    myMap.put(new Date(), "later");

    ObjectMapper mapper = new ObjectMapper();
    System.out.println(mapper.writeValueAsString(myMap));
    // {"Mon Jul 04 11:38:36 MST 2011":"now","Mon Jul 04 11:38:36 MST 2011":"later"}

    SimpleModule module =  
      new SimpleModule("MyMapKeySerializerModule",  
          new Version(1, 0, 0, null));
    module.addKeySerializer(Date.class, new DateAsTimestampSerializer());

    MapType myMapType = TypeFactory.defaultInstance().constructMapType(HashMap.class, Date.class, String.class);

    ObjectWriter writer = new ObjectMapper().withModule(module).typedWriter(myMapType);
    System.out.println(writer.writeValueAsString(myMap));
    // {"1309806289240":"later","1309806289140":"now"}
  }
}

class DateAsTimestampSerializer extends JsonSerializer<Date>
{
  @Override
  public void serialize(Date value, JsonGenerator jgen, SerializerProvider provider) 
      throws IOException, JsonProcessingException
  {
    jgen.writeFieldName(String.valueOf(value.getTime()));
  }
}

最新のジャクソン(2.0.4)の更新:

import Java.io.IOException;
import Java.util.Date;
import Java.util.HashMap;
import Java.util.Map;

import com.fasterxml.jackson.core.JsonGenerator;
import com.fasterxml.jackson.core.JsonProcessingException;
import com.fasterxml.jackson.databind.JsonSerializer;
import com.fasterxml.jackson.databind.ObjectMapper;
import com.fasterxml.jackson.databind.ObjectWriter;
import com.fasterxml.jackson.databind.SerializerProvider;
import com.fasterxml.jackson.databind.module.SimpleModule;
import com.fasterxml.jackson.databind.type.MapType;
import com.fasterxml.jackson.databind.type.TypeFactory;

public class CustomKeySerializerDemo
{
  public static void main(String[] args) throws Exception
  {
    Map<Date, String> myMap = new HashMap<Date, String>();
    myMap.put(new Date(), "now");
    Thread.sleep(100);
    myMap.put(new Date(), "later");

    ObjectMapper mapper = new ObjectMapper();
    System.out.println(mapper.writeValueAsString(myMap));
    // {"2012-07-13T21:14:09.499+0000":"now","2012-07-13T21:14:09.599+0000":"later"}

    SimpleModule module = new SimpleModule();
    module.addKeySerializer(Date.class, new DateAsTimestampSerializer());

    MapType myMapType = TypeFactory.defaultInstance().constructMapType(HashMap.class, Date.class, String.class);

    ObjectWriter writer = new ObjectMapper().registerModule(module).writerWithType(myMapType);
    System.out.println(writer.writeValueAsString(myMap));
    // {"1342214049499":"now","1342214049599":"later"}
  }
}

class DateAsTimestampSerializer extends JsonSerializer<Date>
{
  @Override
  public void serialize(Date value, JsonGenerator jgen, SerializerProvider provider) 
      throws IOException, JsonProcessingException
  {
    jgen.writeFieldName(String.valueOf(value.getTime()));
  }
}
43

いつものように、ブルースの答えはその場で正しいです。

もう1つの考えは、Date値をtimestampsとしてシリアル化するためのグローバル設定があるためです。

_SerializationConfig.Feature.WRITE_DATES_AS_TIMESTAMPS
_

多分それはここにも当てはまるはずです。または、少なくとも標準の ISO-8601 形式をテキストに使用します。主な実用的な問題は、後方互換性の問題です。ただし、私は現在のプレーンなtoString()の使用が効率的でも便利でもない(値を読み返す)ため、非常に役立つとは思えません。

したがって、必要に応じて、機能のリクエストを提出することができます。これは、JacksonによるMapキーの最適ではない処理のように聞こえます。

5
StaxMan

ジャクソン2.0以降(おそらく1.9も)、WRITE_DATE_KEYS_AS_TIMESTAMPSを使用して、この特定の動作を変更できます。

ObjectMapperの使用例:

ObjectMapper m = new ObjectMapper().configure(SerializationFeature.WRITE_DATE_KEYS_AS_TIMESTAMPS, true);

ObjectWriterの場合:

ObjectWriter w = mapper.with(SerializationFeature.WRITE_DATE_KEYS_AS_TIMESTAMPS);
1
Roben