目录

介绍

背景

使用代码

兴趣点


如果您的Json包含日期时间值如“2018-11-23T20:56:05.3117673Z”并且您需要将其作为正确的BsonDateTime值放在BsonDocument中,该怎么办?

介绍

这是我的第一篇文章,最后我发现了一个问题,我认为在这个领域(JSON,DateTime和使用BsonDocument的MongoDb)中有很多人都有值得发表的问题!我很高兴与您分享解决方案!

大多数时候,人们使用Json作为序列化格式在应用程序之间传输数据。MongoDB有内置的序列化器,可以从Json转换为Bson。问题是生成的Bson不会将预期的DateTime 字符串处理为BsonType.DateTime ,而是将其处理为BsonType.String。我想给你一个关于如何构建东西的选项,将Json字符串值(具有给定DateTime格式)正确地反序列化为BsonDateTime值。

背景

MongoDb中内置的序列化程序可以在满足给定格式时处理“Json”中的datetime字符串表示,即:

{"myDate": ISODate("2018-11-23T20:56:05.311Z")
}

但这不是一个有效的Json格式,并且没有Json序列化程序(如Json.NET)能够正确地序列化或反序列化(无需任何修改)。大多数人都希望拥有以下Json格式:

{"myDate": "2018-11-23T20:56:05.311Z"
}

通常,您可以使用以下代码将Json反序列化为Bson:

var bson = BsonSerializer.Deserialize<BsonDocument>(json);

你会看到结果是BsonType.String。

当您检查BsonSerializer源代码时,您将找到以下代码:

public static TNominalType Deserialize<TNominalType>(string json, Action<BsonDeserializationContext.Builder> configurator = null)
{using (var bsonReader = new JsonReader(json)){return Deserialize<TNominalType>(bsonReader, configurator);}
}

所以在幕后,MongoDb使用某种JsonReader,只是解析任何string值为BsonType.String。

这是我们将扩展JsonReader的地方,以便能够正确地解析上面提到的datetime模式,从而产生一个BsonType.DateTime。

使用代码

让我们创建自己的DateTimeAwareJsonReader,只是扩展现有的JsonReader并尝试确定给定string值是否可以是有效的日期时间。但我们要记住,我们希望它尽可能高效。我们知道我们必须调查Json值的每个string表示,无论它是否是datetime。让我们开始吧:

public class DateTimeAwareJsonReader : JsonReader
{public DateTimeAwareJsonReader(string json) : base(json){}public DateTimeAwareJsonReader(TextReader textReader) : base(textReader){}public DateTimeAwareJsonReader(string json, JsonReaderSettings settings) : base(json, settings){}public DateTimeAwareJsonReader(TextReader textReader, JsonReaderSettings settings) : base(textReader, settings){}
}

序列化引擎使用IBsonReader.ReadBsonType()以确定当前正在解析哪种类型,因此我们必须挂钩并在此处进行调查,无论我们是否想告诉引擎给定的值是 string或datetime。所以让我们添加一个重写方法:

private string _currentStringValue;
private BsonDateTime _currentDateTime;public override BsonType ReadBsonType()
{_currentDateTime = null;var currentBsonType = base.ReadBsonType();if (currentBsonType == BsonType.String){var previousState = State;_currentStringValue = ReadString();State = previousState;// date pattern is YYYY-MM-dd, if this pattern is met, // we assume it is a DateTime and try to parse.// kind of like duck typing..if it walks like a duck and talks like a duck, // it must be a duck.. :Dif (_currentStringValue.Length > 9){if (char.IsDigit(_currentStringValue[0]) &&char.IsDigit(_currentStringValue[1]) &&char.IsDigit(_currentStringValue[2]) &&char.IsDigit(_currentStringValue[3]) &&_currentStringValue[4].Equals('-') &&char.IsDigit(_currentStringValue[5]) &&char.IsDigit(_currentStringValue[6]) &&_currentStringValue[7].Equals('-') &&char.IsDigit(_currentStringValue[8]) &&char.IsDigit(_currentStringValue[9])){// looks like a date stringif (DateTime.TryParse(_currentStringValue, out var parsedDateTime)){_currentDateTime = new BsonDateTime(parsedDateTime);CurrentBsonType = BsonType.DateTime; // we have to set also the // current bson type and return it.return BsonType.DateTime;}}}}return currentBsonType;
}

基础的JsonReader将告诉我们,我们的“ myDate”属性是BsonType.String。在这种情况下,我们想要拦截并进一步的调查。为了提高效率,我们只检查至少给定大小的值。我希望序列化datetime的格式至少为“ YYYY-MM-dd”(例如“ 2018-05-02”)。如果您有其他要求,可以根据需要在此处调整逻辑。所以一旦我们在这里涉及到数字,并且所有内容看起来都像是string表示的datetime,我们将告诉序列化器这个值是BsonType.DateTime,否则我们想要回退到序列化器必须决定的情况。

一旦我们认为给定的值是DateTime,我们必须重写另一个方法:

public override long ReadDateTime()
{if (_currentDateTime == null){return base.ReadDateTime();}if (Disposed) { ThrowObjectDisposedException(); }VerifyBsonType("ReadDateTime", BsonType.DateTime);State = BsonReaderState.Type;return _currentDateTime.AsBsonDateTime.MillisecondsSinceEpoch;
}

如果我们的逻辑开始,该字段_currentDateTime将具有解析的值,我们将返回它。

这就是它!您现在有一个可以正确计算datetime 字符串表示的工作解决方案,并告诉MongoDb它将其作为一个DateTime处理。

要测试它,你可以做一个简单的控制台应用程序:

class Program
{public class SomeModel{[JsonProperty("id")]public int Id { get; set; }[JsonProperty("test")]public string Test { get; set; }[JsonProperty("today")]public DateTime Today { get; set; }[JsonProperty("inner")]public SomeModel Inner { get; set; }}static void Main(string[] args){var model = new SomeModel{Id = 1,Test = "Model",Today = DateTime.UtcNow.Date,Inner = new SomeModel{Id = 2,Test = "Inner",Today = DateTime.UtcNow}};var json = JsonConvert.SerializeObject(model);using (var reader = new DateTimeAwareJsonReader(json)){var bson = BsonSerializer.Deserialize<BsonDocument>(reader);Console.WriteLine("Models today property is: {0}", bson["today"].BsonType);}Console.ReadLine();}
}

兴趣点

一旦我弄清楚MongoDb序列化是如何工作的,就很容易构建一个拦截机制来告诉底层的序列化框架datetime应该是什么。我还没有在高效的环境中测试它,但我很快就会做到。更有趣的部分是以一种精益的方式来猜测一个string值是否可以是datetime最有效的方式,而不会过多地干扰MongoDb客户端库。我认为这个解决方案适用于大多数用例。期待您的意见!;-)

原文地址:https://www.codeproject.com/Tips/1268019/MongoDB-Csharp-How-to-Deserialize-a-JSON-Containin

MongoDB C#:如何将包含DateTime的JSON反序列化为正确的BsonDocument DateTime值相关推荐

  1. 是否可以将 json 反序列化为 dynamic 对象?

    咨询区 jswanson: 是否有方法可以将 json 反序列化为 dynamic 对象,如果可以实现的话,那我就可以省去很多冗余类. 回答区 Tom Peplow: 如果你用 Json.NET 的话 ...

  2. JavaScriptSerializer类 对象序列化为JSON,JSON反序列化为对象

    JavaScriptSerializer 类由异步通信层内部使用,用于序列化和反序列化在浏览器和 Web 服务器之间传递的数据.说白了就是能够直接将一个C#对象传送到前台页面成为javascript对 ...

  3. java string 反序列化_如何将java.lang.String的空白JSON字符串值反序列化为null?

    我正在尝试使用简单的JSON反序列化为Java对象.不过,我,让空 字符串 值,java.lang.String属性值.在其余的属性中,空白值将转换为 空 值(这是我想要的). 我的JSON和相关的J ...

  4. java hashmap 实现 序列化_java – Jackson JSON对象映射器反序列化为LinkedHashMap而不是HashMap...

    我有一个有内部地图的POJO.我希望从我的 JSON反序列化为HashMap,但Jackson将内部映射从JSON反序列化为LinkedHashMap.我可以通过将Map的类型从"Map&q ...

  5. python读取多行json_如何在Python中读取包含多个JSON对象的JSON文件?

    所以这是在python中读取JSON文件的标准方法 import json from pprint import pprint with open('ig001.json') as data_file ...

  6. 包含对象的json格式_如何把JSON数据格式转换为Python的类对象?

    JOSN字符串转换为自定义类实例对象 有时候我们有这种需求就是把一个JSON字符串转换为一个具体的Python类的实例,比如你接收到这样一个JSON字符串如下: {"Name": ...

  7. [转载] 包含对象的json格式_如何把JSON数据格式转换为Python的类对象?

    参考链接: python json 12: numpy转换为json数据 JOSN字符串转换为自定义类实例对象 有时候我们有这种需求就是把一个JSON字符串转换为一个具体的Python类的实例,比如你 ...

  8. 在Python中,json.jumps无法直接转译datetime类型的值

    python中,json.jumps无法直接转译datetime类型的值. 最简便的方法可以通过下面这个: json.dumps(results, indent=4, sort_keys=True, ...

  9. 下载网上压缩包(包含多行json)并将其转换为字典的解决方案

      大家好,我是爱编程的喵喵.双985硕士毕业,现担任全栈工程师一职,热衷于将数据思维应用到工作与生活中.从事机器学习以及相关的前后端开发工作.曾在阿里云.科大讯飞.CCF等比赛获得多次Top名次.现 ...

最新文章

  1. C/C++ 读取任意数目的整数
  2. 使用putty上传文件
  3. 学习笔记整理之对象的转型
  4. bootstrap-fileupload-上传文件控件
  5. smooth_L1_loss_layer.cpp:28] Check failed: bottom[0]-channels() == bottom[1]-channels() (12 vs. 84
  6. 最新后端架构师技术图谱
  7. 迈达斯cdn使用说明_快速了解CDN是什么
  8. 关于双网卡电脑无法加入域或失去与域的联系的方法
  9. cs61b实验记录(八)project 3:BearMaps 基于OSM的伯克利地图web应用
  10. ln多少等于2用计算机,ln2(log计算器在线)
  11. 英文科技论文写作中常见的问题和技巧(自用整理)
  12. matlab 矩阵求和计算
  13. 量化策略:如何利用死猫反弹获利?
  14. 使用ColorMatrix简单处理色彩平衡
  15. Rosalind Java| Complementing a Strand of DNA
  16. ATT7022系列STM32F1的HAL库读写操作流程
  17. 用计算机把老式的织布机,电脑织布机的操作怎样操作电脑织布机
  18. 【英语学习】【WOTD】writhe 释义/词源/示例
  19. 动态路由协议—EIGRP
  20. [转] Linux ALSA声卡驱动之三:component、dai、codec以及platform之间的关系

热门文章

  1. big sur 黑苹果_苹果宣布11日再开发布会!自研芯片届时或将发布!
  2. 护肤品APP界面设计模板,可以临摹的UI好素材
  3. 电商设计提升水平,需要优秀的模板素材进行临摹练习!
  4. linux sleeping进程多_Linux下找出吃内存的方法总结
  5. 湖南高工计算机考试,湖南一工教育
  6. mysql 锁设置密码_[转载]mysql锁小结
  7. Go语言学习Day05
  8. Python读写文件说明
  9. 消逝波(表面波)Evanescent Wave
  10. NVIDIA Tesla K40C 和 AMD Firepro W8100 的对比