使用c访问mongodb,需要用到mongodb c driver。c++的driver也是基于c driver封装的。

在使用c driver访问mongodb时,需要与bson打交道,不过c driver访问bson有几点需要注意的,不然会导致报错,或者找不到数据。

迭代器使用后的有效性

在mongodb c driver中,使用bson_iter_find等操作迭代器后,不要再次使用,有几点原因:

  • 如果api报错,比如没找到key,迭代器就会失效,如果继续用,会导致程序崩溃
  • 如果api没报错,找到了key,那么下一次继续使用,查找的value,必须是在当前找到的value后面,如果在迭代器前面,就无法找到。

所以一般操作是在使用迭代器之前,赋值一个局部变量,对这个局部变量进行操作。

官方有如下解释:

The bson_iter_find() function shall advance iter to the first element named key or exhaust all elements of iter. If iter is exhausted, false is returned and iter should be considered invalid.

示例

const bson_t *doc = nullptr;
bson_iter_t iter;
while (mongoc_cursor_next(cursor, &doc))
{
// 从doc初始化获得iter
if (!bson_iter_init(&iter, doc))
{
continue;
}
// 创建一个临时变量的迭代器,用作后续操作
bson_iter_t proiter; // 把iter赋值给proiter,避免iter失效
proiter = iter;
// 使用proiter进行查找
if (bson_iter_find(&proiter, "key1"))
{
...
} // 再次把iter赋值给proiter,进行后续操作
proiter = iter;
if (bson_iter_find(&proiter, "key2"))
{
...
}
...
}

子对象与数组遍历的层次结构

使用mongodb c driver访问数组结构或者子对象时,也需要注意,其中多了一层,如果少进入一层,就拿不到数据。

数组是结构体

{
"key1": "111",
"key2":
[
{
"ip": "127.0.0.1",
"protocal": "tcp"
},
{
"ip": "127.0.0.2",
"protocal": "udp"
}
]
}

如上,有一个json数据,key1是一个字符串,key2是一个结构体数组,当我们遍历key2时,其每个元素也是key-value结构,key是数组索引0、1、2...,value才是数组元素。所以进入key2之后,还要再进入一层,才能拿到ip和protocal数据。

示例

// 找到key2,然后递归一层,下一层数据的迭代器赋值给sub_iter
if (bson_iter_find(&proiter, "key2") && bson_iter_recurse(&proiter, &sub_iter))
{
// 遍历sub_iter的所有元素,这里就是上面说的,key是数组的索引,value是ip和protocal结构体
while (bson_iter_next(&sub_iter))
{
bson_iter_t subsubiter;
// 赋值给pproiter局部变量,避免sub_iter失效
bson_iter_t pproiter = sub_iter;
// 再递归一层,才能拿到ip和protocal
if (bson_iter_recurse(&pproiter, &subsubiter))
{
bson_iter_t ppproiter = subsubiter;
// 找到ip
if (bson_iter_find(&ppproiter, "ip"))
{
}
}
}
}

数组是value(字符串)

"key1":
{
"ip":
[
"192.168.1.10",
"192.168.1.12",
"192.168.1.13",
"192.168.1.17"
]
}

如果数组是字符串呢?实际上是少了一层,但是也不要忘记,找到数组的迭代器,还是需要递归一层才可以拿到数据。

示例

// 找到key1
if (bson_iter_find(&iter, "key1"))
{
// 递归一层,获得key1的下一层ip
bson_iter_recurse(&iter, &subiter);
// 找到ip
if (bson_iter_find(&subiter, "ip"))
{
bson_iter_t piter;
// 递归一层,找到ip下一层的数据
bson_iter_recurse(&subiter, &piter);
// 遍历ip数组内的元素
while(bson_iter_next(&piter))
{
const char *name = bson_iter_utf8 (&piter, NULL);
printf("%s\n", name);
}
}
}

从上面可以看出,mongodb c driver的下一层级的数据遍历,都是先找到对应的key,再递归一层,才能拿到对应的数据。

https://www.mongodb.com/docs/drivers/c/

http://mongoc.org/libbson/current/index.html

http://mongoc.org/

mongodb c driver bson的嵌套访问与层次结构的更多相关文章

  1. MongoDB C Driver使用教程

    MongoDB C Driver使用教程 转载请注明出处http://www.cnblogs.com/oloroso/ 本指南提供简介 MongoDB C 驱动程序. 在 C API 的详细信息,请参 ...

  2. Ignoring Extra Elements in mongoDB C# Driver

    MongoDB删除字段后会报错: Element ... does not match any field or property of class Customer. 需要在实体类增加 [BsonI ...

  3. mongodb .net driver

    1.介绍 The official MongoDB .NET Driver provides asynchronous interaction with MongoDB. Powering the d ...

  4. Mongodb Java Driver 参数配置解析

    要正确使用Mongodb Java Driver,MongoClientOptions参数配置对数据库访问的并发性能影响极大. connectionsPerHost:与目标数据库能够建立的最大conn ...

  5. 给java mongodb 官方driver 增加bean 操作

      mongodb官方的java driver不支持直接插入java bean,只能使用DbObject的Key,Value形式进行insert,update,(c# mongodb官方driver类 ...

  6. MongoDB Java Driver操作指南

    MongoDB为Java提供了非常丰富的API操作,相比关系型数据库,这种NoSQL本身的数据也有点面向对象的意思,所以对于Java来说,Mongo的数据结构更加友好. MongoDB在今年做了一次重 ...

  7. windows平台下安装、编译、使用mongodb C++ driver

    本博客将记录在Win8.1 ,VS2013环境下编译.配置mongodb C++ driver的流程. 1.下载预备 下载Boost:http://sourceforge.net/projects/b ...

  8. mongodb c++ driver(2.53)windows编译

    编译环境: (1) 下载python2.7, 使用x86_32位,因为scons只有32位安装包可用: (2) 下载scons2.3.0,The current production release ...

  9. MongoDB C Driver and APIinstances linux MongoDB安装配置

    <一,linux平台MongoDB安装配置>在这我们使用的Centos6 yum部署的,你想搞编译,自个干!

  10. MongoDB C driver API continues

    开篇前 <1,mongoc_init() func> mongoc_init() Synopsis void mongoc_init (void); Description This fu ...

随机推荐

  1. 更快的训练和推理: 对比 Habana Gaudi®2 和英伟达 A100 80GB

    通过本文,你将学习如何使用 Habana Gaudi2 加速模型训练和推理,以及如何使用 Optimum Habana 训练更大的模型.然后,我们展示了几个基准测例,包括 BERT 预训练.Stabl ...

  2. 【解惑】介绍三大数据库的with语句的写法及使用场景

    WITH 子句通常被称为 "Common Table Expressions"(CTE),俗称内存临时表,当使用 WITH 语句时,应注意具体的数据库版本和支持情况.以下是对 My ...

  3. [golang]使用tail追踪文件变更

    简介 借助 github.com/hpcloud/tail ,可以实时追踪文件变更,达到类似shell命令tail -f的效果. 示例代码 以下示例代码用于实时读取nginx的access.log日志 ...

  4. 合宙ESP32C3使用PlatformIO开发点亮ST7735S

    开发背景 模块使用的合宙的ESP32-C3(经典款) 购买连接 CORE ESP32核心板是基于乐鑫ESP32-C3进行设计的一款核心板,尺寸仅有21mm*51mm,板边采用邮票孔设计,方便开发者在不 ...

  5. 文心一言 VS 讯飞星火 VS chatgpt (76)-- 算法导论7.3 1题

    一.如果用go语言,为什么我们分析随机化算法的期望运行时间,而不是其最坏运行时间呢? 文心一言: 在分析算法的性能时,期望运行时间是一个重要的指标,它描述了算法在平均情况下的表现.期望运行时间考虑了所 ...

  6. MyBatis Mapper映射处理CLOB和BLOB类型

    ​Mybatis的MapperXML映射文件应该处理数据库字段类型为CLOB和BLOB类型的数据呢?首先我们先看下CLOB和BLOB这两种数据类型的介绍. 介绍 使用Mybatis时涉及到两种特殊类型 ...

  7. 《数据结构-C语言》单链表

    @ 目录 单链表 结构定义 初始化 建立 清空 求表长 判断是否为空表 取值 查找 插入 删除 销毁 遍历打印 测试 单链表 结构定义 #include <stdio.h> #includ ...

  8. IOS苹果应用IPA重签名软件手机版(苹果重签名,企业签名,安卓苹果平台,时间控制)

    软件简介 IOS苹果应用IPA重签名软件手机版,可以在安卓或者苹果手机上,苹果应用IPA文件重新签名,无需MAC苹果电脑和配置XCODE开发环境,便可以直接对IPA文件进行签名,签名在本地进行,不消耗 ...

  9. 通过 Haproxy 实现 ss 负载均衡

    介绍 缺点:所有的SS的加密方式和密码必须一致 介绍:HAProxy是一个使用C语言编写的自由及开放原始码软件,其提供高可用性.负载均衡,以及基于TCP和HTTP的应用程序代理. 安装Haproxy ...

  10. 「codeforces - 585E」Present for Vitalik the Philatelist

    link. 设 \(\displaystyle f(x) = \# S', s.t. S' \subseteq S, S' \neq \varnothing, \gcd(S') = x\),\(g(x ...