我正试图使用Json.NET串行化一个类,以便通过http请求进行传输。这是一个客户端-服务器测试程序,它共享一些常见的类文件。这些文件是一个接口(ITestCase
)和该接口的实现(TestPicture
、TestVideo
)。
在同一个应用程序中串行化和取消串行化下面的testCase
可以很好地工作,可能是因为Json.NET代码都包含在一个程序集中。当我串行化testCase
,将其发送到服务器,然后尝试取消串行化时,我得到错误
"Error resolving type specified in JSON 'com.test.testcases.TestPicture, Graphics Tester'. Path '$type', line 2, position 61"
具有JsonSerializationException类型的内部异常,消息
"Could not load assembly 'Graphics Tester'."
在Json.NET文档中,生成Json时,$type值为"$type": "Newtonsoft.Json.Samples.Stockholder, Newtonsoft.Json.Tests"
。第二个参数似乎是按名称空间引用相关类,而不是像在我的实例中那样按项目名称(即Graphics Tester)引用。
既然两个项目共享必需的类,并且文件位于同一命名空间中,那么Json.NET为什么要查找程序集而不是类本身呢?
下面是我的代码的骨架;细节被忽略了,因为它们对解决问题没有帮助。显示的是接口,以及该接口的两个实现。还显示了串行化和去串行化操作,以及由此产生的Json。
ITestCase testCase = new TestPicture("test.jpg")
string json = JsonConvert.SerializeObject(testCase, Formatting.Indented, new JsonSerializerSettings
{
TypeNameHandling = TypeNameHandling.Objects,
});
ITestCase instance = JsonConvert.DeserializeObject<ITestCase>(json, new JsonSerializerSettings
{
TypeNameHandling = TypeNameHandling.Objects,
});
//value in variable json after serialisation
//{
// "$type": "com.test.testcases.TestPicture, Graphics Tester",
// "filename": "test.jpg",
// "testPoints": null
//}
单个类文件:
namespace com.test.testcases
{
interface ITestCase
{
void Run();
bool Verify();
}
}
namespace com.test.testcases
{
class TestPicture : ITestCase {}
}
namespace com.test.testcases
{
class TestVideo : ITestCase {}
}
我的解决方案:
有一个比我预想的更简单的解决方案。它不是最佳的,但它确实有效。我会把它更多地归类为变通方法或破解方法。通过更改项目属性并在两个项目中将程序集名称设置为相同,它将查找项目中已经包含的类,从而创建Json字符串指定的对象。
$type
对完全限定的类型名进行编码,如果客户端和服务器不使用同一项目来定义这些类型(即,如果每个类型都自己定义类型,而不是引用公共程序集),则完全限定的类型名在客户端和服务器之间会有所不同。
在您的情况下,在客户端中,完全限定的名称是com.test.testcases.TestPicture, Graphics Tester
,但当它到达服务器时,服务器找不到任何名为Graphics Tester
的程序集。
您可以通过以下两种方法之一解决此问题:
- 为可序列化类型定义一个通用程序集,并从客户端和服务器引用它
- 在客户端和服务器中分别定义类型,并通过向JsonSerializerSettings.Binder提供自定义SerializationBinder来自定义Json.NET的类型解析,您可以在其中实现自己的逻辑