使用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. 【教程】青少年CTF机器人使用教程

    前言 本期教程适用于版本号为2.0.1-Beta的青少年CTF机器人,其他版本可能与当前版本不同. 由于之前版本的机器人重构,所以我们细化了本次的机器人逻辑,并且对机器人的功能进行了一些升级. 机器人 ...

  2. centos7离线安装harbor

    前言 harbor是一个docker私有仓库,基于docker官方的registry,提供GUI.权限控制.项目管理等功能. 安装harbor前,需要先安装docker和docker-compose ...

  3. jsp+servlet实战项目

    第一步:新建maven项目,项目中添加dao,entity,service,servlet,util包第二步:导入依赖 第三步:数据库建表 第四步:entity实体包(疯转) 第五步:在util工具包 ...

  4. 【故障公告】多年的故障老朋友又来了:数据库服务器 CPU 100%

    数据库服务器 CPU 100% 问题几乎每年都要来几次,从来都不事先打一声招呼,今年的第2次在我们正忙着会员救园的时候来了. 今天 13:35 首先收到我们自己的异常告警通知: Execution T ...

  5. Java 设计模式实战系列—单例模式

    本文首发公众号:小码A梦 单例模式是设计模式中最简单一个设计模式,该模式属于创建型模式,它提供了一种创建实例的最佳方式. 单例模式的定义也比较简单:一个类只能允许创建一个对象或者实例,那么这个类就是单 ...

  6. C++笔记(自用)

    <Effective C++> 条款11 在operator=中处理"自我赋值" 自我赋值 证同测试: if(this==&rhs)return*this; 影 ...

  7. 【题解】Educational Codeforces Round 142(CF1792)

    没有手速,再加上被 E 卡了,废掉了. A.GamingForces 题目描述: Monocarp 正在玩电脑游戏.他打算杀死 \(n\) 个怪兽,第 \(i\) 个的血量为 \(h_i\). Mon ...

  8. 「luogu - P4126」「ahoi 2009」最小割

    link. 也许题不错,反正有点降智- 先给结论,在 \[V_N=V \\ E_N=E \\ c(x,y)=w(x,y) \] 的流网络中: 可行边:在增广完的 induced subgraph 中, ...

  9. xftp 7必须更新最新版本怎么解决

    下载可以查看16进制的软件: Sublime Text 运行XFTP7 双击打开是:这样的 解决方案 用Sublime Text进行打开nslicense.dll, 打开之后查找"0f88 ...

  10. 数据泵(impdb)导入Oracle分片的数据库dump文件

    数据泵(impdb)导入Oracle数据库 一.sqlplus登录目标数据库,创建导入的目录路径 #该目录要在导入的数据库本机建立,如果是docker就在容器内部创建 create directory ...