使用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. 王道oj/problem12(动态申请内存存储数组)

    网址:http://oj.lgwenda.com/problem/12 思路:用输入的整型创建对应数组,用scanf消除换行键: 用gets()输入语句并输出,再释放. 代码: #define _CR ...

  2. 【Azure K8S | AKS】在不丢失文件/不影响POD运行的情况下增加PVC的大小

    问题描述 在前两篇文章中,创建了Disk + PV + PVC + POD 方案后,并且进入POD中增加文件. [Azure K8S | AKS]在AKS集群中创建 PVC(PersistentVol ...

  3. Jmeter:"An error occurred: Can't connect to X11 window server using 'lacalhost:12.0' as the value of the display variable." 解决办法

    做各种不同项目的性能测试,都需要在项目本地压测服务器配置jmeter,需要时还要调出jmeter图形化界面来调试jmeter脚本. 标题中的问题遇过多次,这次做个记录. 1. 启动jmeter报错 在 ...

  4. [Pwn之路]根据所给库,获得远程同环境——使用patchelf的正确姿势

    原文:https://www.freebuf.com/sectool/366854.html 存自己这里方便看. 0x00 前言 如何修改本地pwn文件和题目所给环境一致,从而进行调试,这是从学习堆开 ...

  5. c++算法:二分

    算法中,有一种比线性查找算力费得更少的一种算法思想,叫"分治",今天讲的是分治里的二分查找: 借助 (low+high)/2公式,找到搜索区域内的中间元素.图 1 中,搜索区域内中 ...

  6. Shiro配置类中的各个配置项浅谈

    背景: 上文中在落地实践时,对Shiro进行了相关的配置,并未对其含义作用进行详细学习,本章将进一步详解其作用含义. Shiro配置类中的各个配置项的作用: @Bean public Security ...

  7. java循环自动生成简单图片

    import java.awt.*; import java.awt.font.FontRenderContext; import java.awt.geom.Rectangle2D; import ...

  8. MySQL的字段数量以及长度限制

    一.InnoDB行格式 行格式 紧凑的存储特性 增强的可变长度列存储 大型索引键前缀支持 压缩支持 支持的表空间类型 REDUNDANT N N N N system, file-per-table, ...

  9. Modbus转Profinet网关改写变频器运行状态在1200PLC程序控制实例

    兴达易控Modbus转Profinet网关将丹佛斯变频器接入西门子 1200PLC 配置案例 案例简介: 本案例是兴达易控Modbus转Profinet网关连接丹佛斯变频器在西门子 1200PLC程序 ...

  10. Quantitative Relationship Induction

    数量关系是指事物之间的数值或数量之间的相互关系(+.-.*./). 数量关系描述各种量的变化和相互关系.数量关系可以包括数值的比较.增减.比例.百分比.平均值等方面. 在数学中,数量关系可以通过代数方 ...