- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我们有一种情况,我们使用 JSON 序列化 EF 模型以用于数据同步。为了使同步正常工作,我们需要模型的表名。这并不难,我们已经有了代码。主要问题是在 JSON 中传输该数据。
例如,假设我们有以下模型。
public class Foo
{
// ...
public virtual ICollection<Bar> Bars { get; set; }
}
public class Bar
{
// ...
public virtual ICollection<FooBar> FooBars { get; set; }
}
public class FooBar
{
// ...
}
我们通过 include 拉下所有嵌套项,然后将其序列化。问题是,我们需要将实体的表名作为元数据插入到 JSON 中,而不是将其添加到模型本身。
例如,在上面的场景中,JSON 看起来像
{
"__tableName": "Foos",
// ...
"Bars": [
{
"__tableName": "Bars"
// ...
"FooBars": [
{
"__tableName": "FooBars"
// ...
}
]
}
]
}
我认为 JSON.Net 中的自定义序列化器是最好的方式,但要么我没有插入正确的位置,要么它们没有按照我认为的方式工作。
我试图制作自定义 JsonConverter
,因为这似乎是处理自定义序列化场景的默认方式。但是,它似乎只在要序列化的基础对象上调用,而不是任何子对象。
我需要在某个地方插入 JSON.Net 才能实际放入此元数据吗?我在这个主题上找到的几乎所有内容都指向 JsonConverter,但我不确定在这种情况下它是否真的能满足我的需要。
有人认为我是将对象加载到 JsonConverter
中的 JObject
中,然后自己遍历模型树并根据需要插入键,但我希望得到一些东西比那更优雅一点。
谢谢。
最佳答案
尽管 JsonConverter
在这里似乎是合适的选择,但实际上对于此类问题,它在实践中效果不佳。问题是您希望以编程方式将转换器应用于一组广泛的类,而这些类可以包含使用相同转换器的其他类。因此,您可能会遇到转换器中递归循环的问题,您需要解决这个问题。可以做到,但可能会有点困惑。
幸运的是,对于这种情况有更好的选择。您可以将自定义 IContractResolver
与 IValueProvider
结合使用,以将 __tableName
属性插入到每个具有表名的对象的 JSON 中。解析器负责检查特定对象类型是否具有关联的表名,如果有,则为该类型设置额外的属性。值提供程序仅在被询问时返回表名。
这是您需要的代码:
class TableNameInsertionResolver : DefaultContractResolver
{
private Dictionary<string, string> tableNames;
public TableNameInsertionResolver(Dictionary<string, string> tableNames)
{
this.tableNames = tableNames;
}
protected override IList<JsonProperty> CreateProperties(Type type, MemberSerialization memberSerialization)
{
IList<JsonProperty> props = base.CreateProperties(type, memberSerialization);
// If there is an associated table name for this type,
// add a virtual property that will return the name
string tableName;
if (tableNames.TryGetValue(type.FullName, out tableName))
{
props.Insert(0, new JsonProperty
{
DeclaringType = type,
PropertyType = typeof(string),
PropertyName = "__tableName",
ValueProvider = new TableNameValueProvider(tableName),
Readable = true,
Writable = false
});
}
return props;
}
class TableNameValueProvider : IValueProvider
{
private string tableName;
public TableNameValueProvider(string tableName)
{
this.tableName = tableName;
}
public object GetValue(object target)
{
return tableName;
}
public void SetValue(object target, object value)
{
throw new NotImplementedException();
}
}
}
要将其插入序列化管道,请创建一个 JsonSerializerSettings
实例并将 ContractResolver
属性设置为自定义解析器的一个实例。然后将设置传递给序列化程序。就是这样;它应该“正常工作”。
这是一个演示:
class Program
{
static void Main(string[] args)
{
Foo foo = new Foo
{
Id = 1,
Bars = new List<Bar>
{
new Bar
{
Id = 10,
FooBars = new List<FooBar>
{
new FooBar { Id = 100 },
new FooBar { Id = 101 }
}
},
new Bar
{
Id = 11,
FooBars = new List<FooBar>
{
new FooBar { Id = 110 },
new FooBar { Id = 111 },
}
}
}
};
// Dictionary mapping class names to table names.
Dictionary<string, string> tableNames = new Dictionary<string, string>();
tableNames.Add(typeof(Foo).FullName, "Foos");
tableNames.Add(typeof(Bar).FullName, "Bars");
tableNames.Add(typeof(FooBar).FullName, "FooBars");
JsonSerializerSettings settings = new JsonSerializerSettings();
settings.ContractResolver = new TableNameInsertionResolver(tableNames);
settings.Formatting = Formatting.Indented;
string json = JsonConvert.SerializeObject(foo, settings);
Console.WriteLine(json);
}
}
public class Foo
{
// ...
public int Id { get; set; }
public virtual ICollection<Bar> Bars { get; set; }
}
public class Bar
{
// ...
public int Id { get; set; }
public virtual ICollection<FooBar> FooBars { get; set; }
}
public class FooBar
{
// ...
public int Id { get; set; }
}
输出:
{
"__tableName": "Foos",
"Id": 1,
"Bars": [
{
"__tableName": "Bars",
"Id": 10,
"FooBars": [
{
"__tableName": "FooBars",
"Id": 100
},
{
"__tableName": "FooBars",
"Id": 101
}
]
},
{
"__tableName": "Bars",
"Id": 11,
"FooBars": [
{
"__tableName": "FooBars",
"Id": 110
},
{
"__tableName": "FooBars",
"Id": 111
}
]
}
]
}
fiddle :https://dotnetfiddle.net/zG5Zmm
关于c# - 在 EF 模型的 JSON 序列化期间向嵌套对象添加属性,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/33040690/
我的一位教授给了我们一些考试练习题,其中一个问题类似于下面(伪代码): a.setColor(blue); b.setColor(red); a = b; b.setColor(purple); b
我似乎经常使用这个测试 if( object && object !== "null" && object !== "undefined" ){ doSomething(); } 在对象上,我
C# Object/object 是值类型还是引用类型? 我检查过它们可以保留引用,但是这个引用不能用于更改对象。 using System; class MyClass { public s
我在通过 AJAX 发送 json 时遇到问题。 var data = [{"name": "Will", "surname": "Smith", "age": "40"},{"name": "Wil
当我尝试访问我的 View 中的对象 {{result}} 时(我从 Express js 服务器发送该对象),它只显示 [object][object]有谁知道如何获取 JSON 格式的值吗? 这是
我有不同类型的数据(可能是字符串、整数......)。这是一个简单的例子: public static void main(String[] args) { before("one"); }
嗨,我是 json 和 javascript 的新手。 我在这个网站找到了使用json数据作为表格的方法。 我很好奇为什么当我尝试使用 json 数据作为表时,我得到 [Object,Object]
已关闭。此问题需要 debugging details 。目前不接受答案。 编辑问题以包含 desired behavior, a specific problem or error, and the
我听别人说 null == object 比 object == null check 例如: void m1(Object obj ) { if(null == obj) // Is thi
Match 对象 提供了对正则表达式匹配的只读属性的访问。 说明 Match 对象只能通过 RegExp 对象的 Execute 方法来创建,该方法实际上返回了 Match 对象的集合。所有的
Class 对象 使用 Class 语句创建的对象。提供了对类的各种事件的访问。 说明 不允许显式地将一个变量声明为 Class 类型。在 VBScript 的上下文中,“类对象”一词指的是用
Folder 对象 提供对文件夹所有属性的访问。 说明 以下代码举例说明如何获得 Folder 对象并查看它的属性: Function ShowDateCreated(f
File 对象 提供对文件的所有属性的访问。 说明 以下代码举例说明如何获得一个 File 对象并查看它的属性: Function ShowDateCreated(fil
Drive 对象 提供对磁盘驱动器或网络共享的属性的访问。 说明 以下代码举例说明如何使用 Drive 对象访问驱动器的属性: Function ShowFreeSpac
FileSystemObject 对象 提供对计算机文件系统的访问。 说明 以下代码举例说明如何使用 FileSystemObject 对象返回一个 TextStream 对象,此对象可以被读
我是 javascript OOP 的新手,我认为这是一个相对基本的问题,但我无法通过搜索网络找到任何帮助。我是否遗漏了什么,或者我只是以错误的方式解决了这个问题? 这是我的示例代码: functio
我可以很容易地创造出很多不同的对象。例如像这样: var myObject = { myFunction: function () { return ""; } };
function Person(fname, lname) { this.fname = fname, this.lname = lname, this.getName = function()
任何人都可以向我解释为什么下面的代码给出 (object, Object) 吗? (console.log(dope) 给出了它应该的内容,但在 JSON.stringify 和 JSON.parse
我正在尝试完成散点图 exercise来自免费代码营。然而,我现在只自己学习了 d3 几个小时,在遵循 lynda.com 的教程后,我一直在尝试确定如何在工具提示中显示特定数据。 This code
我是一名优秀的程序员,十分优秀!