- 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/
我正在阅读 Python 文档以真正深入了解 Python 语言,并遇到了 filter 和 map 函数。我以前使用过过滤器,但从未使用过映射,尽管我在 SO 上的各种 Python 问题中都见过这
当我尝试打印 BST 的级别顺序时,这个问题提示了我。 这是一个 Pre-Order Sequence: 4, 1, 2, 3, 5, 6, 7, 8 In_order Sequence : 1, 2
我的代码在 main(序列测试;)的第一行出现错误,指出它是对 sequence::sequence() 的 undefined reference 。我无法更改 main 中的代码。有谁知道我该如何
这可能很简单,但我在通常的 latex 指南中找不到任何相关内容。在这句话中: {\em hello\/} “\/”的目的是什么? 最佳答案 这就是所谓的斜体校正。其目的是确保斜体文本后有适当的间距。
当我从 Postgresql 表中删除所有记录,然后尝试重置序列以在插入时开始一个编号为 1 的新记录时,我得到不同的结果: SELECT setval('tblname_id_seq', (SELE
在版本10.0.3中,MariaDB引入了一种称为序列的存储引擎。 其ad hoc为操作生成整数序列,然后终止。 该序列包含正整数,以降序或升序排列,并使用起始,结束和递增值。 它不允许在多个查询中
如何在 Groovy 中获取给定数字的序列,例如: def number = 169 // need a method in groovy to find the consecutive number
基本上,如果这是 .NET,它看起来像这样: ISomething { string A { get; } int B { get; } } var somethings = new List
说以下代码部分(同一块): A <= 1 A <= 2 变量 A 总是被赋值为 2 吗?还是会出现竞争条件并分配 1 或 2? 我对非阻塞赋值的理解是,由硬件在 future 分配变量 A,因此它可能
在运行 WiX 设置时,我正在寻找操作列表及其顺序。不知何故,官方网站似乎没有提供任何信息。 基本问题是我想正确安排我的自定义操作。通常我需要使用 regsvr32.exe 注册一个 DLL,而这只能
F#初学者在这里 我想创建一个类型,它是具有至少一个元素的另一种具体类型(事件)的序列。任何其他元素都可以在以后随时添加。通常在 C# 中,我会创建一个具有私有(private) List 和公共(p
作为构建过程和不断发展的数据库的一部分,我试图创建一个脚本,该脚本将删除用户的所有表和序列。我不想重新创建用户,因为这将需要比所允许的更多的权限。 我的脚本创建了一个过程来删除表/序列,执行该过程,然
我想恢复两个向量的第一个日期和相同向量的第二个日期之间的日期序列,.... 这是一个例子: dates1 = as.Date(c('2015-10-01', '2015-03-27', '2015-0
这个问题已经有答案了: sql ORDER BY multiple values in specific order? (12 个回答) 已关闭 9 年前。 我有一个 sql 语句,我想要ORDER
我想恢复两个向量的第一个日期和相同向量的第二个日期之间的日期序列,.... 这是一个例子: dates1 = as.Date(c('2015-10-01', '2015-03-27', '2015-0
在用java编写代码时,我需要用“],[”分割字符串。下面是我的代码。 try (BufferedReader reader = new BufferedReader(new InputStreamR
这个问题已经有答案了: Project Euler Question 14 (Collatz Problem) (8 个回答) 已关闭 9 年前。 我正在尝试查找数字的 Collatz 序列。以下
我有一个例程函数process_letter_location(const char& c, string &word)。 在我的 main 中,我声明了一系列字符串变量,如下所示: string s
我需要找到最长的多米诺骨牌链,给定一组 12 个随机挑选的多米诺骨牌。我已经递归地生成了多米诺骨牌的所有可能性(使用 0 到 12 的面值有 91 种可能性)。多米诺骨牌由一 block “砖 blo
我有这个数据结构 Seq,它继承了类 vector 但有一些额外的功能。使用这个数据结构 Seq 我有这个预定义的数据结构: typedef Seq > MxInt2d; 我现在想要一个包含多个 Mx
我是一名优秀的程序员,十分优秀!