ICode9

精准搜索请尝试: 精确搜索
首页 > 编程语言> 文章详细

c# – Cosmos DB – CreateDocumentQuery不反序列化抽象类型

2019-07-10 17:05:25  阅读:189  来源: 互联网

标签:json c azure-cosmosdb


我试图让Cosmos DB .NET SDK v1.19.1使用Json.net序列化设置自动将对象反序列化为正确的类型.除了查询文档时,这似乎工作正常.以下面的代码为例:

    public abstract class Shape
    {
        public int Area { get; set; }
    }

    public class Square : Shape { }
    public class Triangle : Shape { }

    public class Entity
    {
        public string Id { get; set; }
        public string Name { get; set; }
        public Shape Shape { get; set; }
    }

    static void Main(string[] args)
    {
        JsonConvert.DefaultSettings = () => new JsonSerializerSettings()
        {
            ContractResolver = new CamelCasePropertyNamesContractResolver(),
            TypeNameHandling = TypeNameHandling.Auto
        };
        var database = "db";
        var collection = "coll";
        var client = new DocumentClient(new Uri("https://docdburi.documents.azure.com:443/"), "supersecretkey", JsonConvert.DefaultSettings());

        var entity = new Entity() { Id = "testid", Name = "John Doe", Shape = new Square() { Area = 5 } };
        var doc = client.CreateDocumentAsync(UriFactory.CreateDocumentCollectionUri(database, collection), entity).Result;
        entity = client.ReadDocumentAsync<Entity>(UriFactory.CreateDocumentUri(database, collection, entity.Id)).Result;

        // all good so far ... next line throws serialization exception on abstract shape type
        var result = client.CreateDocumentQuery<Entity>(UriFactory.CreateDocumentCollectionUri(database, collection), new FeedOptions() { MaxItemCount = 1 })
            .Where(x => x.Id == entity.Id).AsDocumentQuery()
            .ExecuteNextAsync<Entity>().Result;

该文档在Cosmos DB中创建,其形状上的$type属性符合预期,并且检索文档工作正常.只有当我尝试查询抛出异常的文档时.有关如何使其工作的任何想法?

关于如何最好地处理抽象类型的任何其他建议?我有一个相当深的对象图,有几层抽象类型.

解决方法:

这个助手通常对我有用:

public abstract class SerializableObject<T>
{    
    public static T FromJObject(JObject jObject) => 
         Parse($"{jObject}");

    public static T Parse(string json) =>
        JsonConvert.DeserializeObject<T>(json, 
            new JsonSerializerSettings
            {
                TypeNameHandling = TypeNameHandling.Objects
            });

    public JObject ToJObject() => JObject.Parse(ToJson());

    public string ToJson() =>
        JsonConvert.SerializeObject(this, Formatting.Indented, 
            new JsonSerializerSettings
            {
                TypeNameHandling = TypeNameHandling.Objects
            });
}

现在继承你的Entity类:

public class Entity : SerializableObject<Entity>
{
    public string Id { get; set; }
    public string Name { get; set; }
    public Shape Shape { get; set; }
}

并尝试使用JObject以某种方式查询它:

var result = client.CreateDocumentQuery<JObject>(
    UriFactory.CreateDocumentCollectionUri(database, collection), 
    new FeedOptions() { MaxItemCount = 1 })
        .Where(x => x.Id == entity.Id)
        .AsEnumerable()
        .Select(Entity.FromJObject)
        .FirstOrDefault();

标签:json,c,azure-cosmosdb
来源: https://codeday.me/bug/20190710/1426236.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有