如何将Gson TypeAdapter与Mockito嘲笑的对象一起使用?

时间:2017-09-10 15:20:57

标签: java unit-testing gson mockito

我想通过TypeAdapter模拟一个传入Gson的对象:

@RunWith(MockitoJUnitRunner.class)
public class SceneExporterTest {

@Test
public void testWriter() {
    List<SceneObject> sceneObjects = mockSceneObjects();
    Gson gson = new GsonBuilder().registerTypeAdapter(SceneObject.class, new SceneExporter()).create();

    String s = gson.toJson(sceneObjects); //This method ends up with an exception.
}

private List<SceneObject> mockSceneObjects() {
    List<SceneObject> sceneObjects = new LinkedList<>();
    for (int i = 0; i < 50; i++) {
        sceneObjects.add(mockSceneObject(i));
    }
    return sceneObjects;
}

private SceneObject mockSceneObject(int i) {
    SceneObject sceneObject = mock(SceneObject.class);
    //...
    return sceneObject;
}

}

我的类型适配器:

public class SceneExporter extends TypeAdapter<SceneObject> {

    @Override
    public void write(JsonWriter out, SceneObject value) throws IOException {
        out.name("position");
        out.value(toValue(value.getPosition()));
        out.name("scale");
        out.value(toValue(value.getScale()));
        out.name("rotation");
        out.value(toValue(value.getRotation()));
    }

    @Override
    public SceneObject read(JsonReader in) throws IOException {
        return null;
    }
}

但我最终会遇到这样的例外:

java.lang.UnsupportedOperationException: Attempted to serialize java.lang.Class: com.editor.api.scene.objects.SceneObject. Forgot to register a type adapter?

Scene对象是非常重的对象,我不想在测试中正常对它进行即时通信。那么有可能只是嘲笑它吗?我也不想使用间谍。

1 个答案:

答案 0 :(得分:3)

以这种方式创建的SceneObject实例的运行时类型:mock(SceneObject.class)SceneObject$MockitoMock$<SOMEID>

以这种方式创建的SceneObject实例的运行时类型:new SceneObject()SceneObject

因此,当Gson的TypeAdapterRuntimeTypeWrapper在其上下文中为模拟对象查找已注册的TypeAdapter时,由于SceneExporter已针对SceneObject.class进行了注册,因此无法找到。

真正说的是,模拟的SceneObject不是类型 SceneObject(它实际上是SceneObject的子类)因此Gson会找不到你定制的适配器。

如果您声明Gson这样......

Gson gson = new GsonBuilder().registerTypeAdapter(mock(SceneObject.class).getClass(), new SceneExporter()).create();

...然后找到您的定制类型适配器,但该注册看起来“关闭”不是吗?这感觉就像是一个仅限测试的专业化。

如果您确实必须模拟SceneObject,那么我认为您需要注册一个inheritance aware type adapter,但如果您这样做只是为了支持此测试用例,那么感觉就像仅限测试的限制渗入“主要”源树。所以,也许最简单的解决方案是:

  • 创建SceneObject
  • 的真实实例
  • 使用Mockito Spy删除SceneObject
  • 中调用的特定SceneExporter个getter